Vijos 1451 圆环取数 【区间DP】

背景

小K攒足了路费来到了教主所在的宫殿门前,但是当小K要进去的时候,却发现了要与教主守护者进行一个特殊的游戏,只有取到了最大值才能进去Orz教主……

描述

守护者拿出被划分为n个格子的一个圆环,每个格子上都有一个正整数,并且定义两个格子的距离为两个格子之间的格子数的最小值。环的圆心处固定了一个指针,一开始指向了圆环上的某一个格子,你可以取下指针所指的那个格子里的数以及与这个格子距离不大于k的格子的数,取一个数的代价即这个数的值。指针是可以转动的,每次转动可以将指针由一个格子转向其相邻的格子,且代价为圆环上还剩下的数的最大值。

现在对于给定的圆环和k,求将所有数取完所有数的最小代价。

格式

输入格式

输入文件cirque.in的第1行有两个正整数n和k,描述了圆环上的格子数与取数的范围。

第2行有n个正整数,按顺时针方向描述了圆环上每个格子上的数,且指针一开始指向了第1个数字所在的格子。

所有整数之间用一个空格隔开,且不超过10000。

输出格式

输出文件cirque.out仅包括1个整数,为取完所有数的最小代价。

样例1

样例输入1

6 1
4 1 2 3 1 3

样例输出1

21

限制

对于20%的数据,n≤10,k≤3;
对于40%的数据,n≤100,k≤10;
对于60%的数据,n≤500,k≤20;
对于100%的数据,n≤2000,k≤500;

时限1s。

提示


如上图所示,第一步不转动指针,取走4、3两个数,代价为7;
第2步指针顺时针转动2格,圆环上最大数为3,代价为6,取走1、2、3两个数,代价为6;
第3步指针顺时针转动1格,代价为1,取走剩下的一个数1,代价为1;
最小代价为7+6+6+1+1=21。

题目链接:

  https://www.vijos.org/p/1451

题目大意:

  N个数顺序排成一个圈,指针指向第一个数,目标是把所有数取走(取后数字消失但是位置依然保留)。每次可以同时取指针左右各k格(含指针)

  取一个数的费用是数本身的大小,移动指针的费用是每移动一格花费目前剩余数字最大的。问最小花费。

题目思路:

  【区间DP】

  根据规则可以发现,取数字的花费是固定的 sum(1~n),所以只用考虑移动的费用。并且移动过程中肯定是边移边取,所以未取的数字都集中在中间。

  首先一开始指针在1的时候肯定把左右两边的k个都取了。接着就破环成链,k+2~n-k是未取的数字。

  dp[i][j]表示从i到j区间全部取完的最小花费,且此时指针指向i-1(dp[i][j]和dp[j][i]方向不同)

  转移方程分为两个,一个是顺着原来的方向移动一格,另一个是移动到另一边,二者取最优。

  预处理出区间最大值即可(nlogn或n2

 1 //
 2 //by coolxxx
 3 /*
 4 #include<iostream>
 5 #include<algorithm>
 6 #include<string>
 7 #include<iomanip>
 8 #include<map>
 9 #include<stack>
10 #include<queue>
11 #include<set>
12 #include<bitset>
13 #include<memory.h>
14 #include<time.h>
15 #include<stdio.h>
16 #include<stdlib.h>
17 #include<string.h>
18 #include<math.h>
19 //#include<stdbool.h>
20 #define min(a,b) ((a)<(b)?(a):(b))
21 #define max(a,b) ((a)>(b)?(a):(b))
22 #define swap(a,b) ((a)^=(b),(b)^=(a),(a)^=(b))
23 */
24 #include<bits/stdc++.h>
25 #pragma comment(linker,"/STACK:1024000000,1024000000")
26 #define abs(a) ((a)>0?(a):(-(a)))
27 #define lowbit(a) (a&(-a))
28 #define sqr(a) ((a)*(a))
29 #define mem(a,b) memset(a,b,sizeof(a))
30 #define eps (1e-8)
31 #define J 10000
32 #define mod 1000000007
33 #define MAX 0x7f7f7f7f
34 #define PI 3.14159265358979323
35 #define N 2004
36 using namespace std;
37 typedef long long LL;
38 double anss;
39 LL aans;
40 int cas,cass;
41 int n,m,lll,ans;
42 int a[N],sum[N];
43 int maxx[N][N],dp[N][N];
44
45 int main()
46 {
47     #ifndef ONLINE_JUDGE
48     freopen("1.txt","r",stdin);
49 //    freopen("2.txt","w",stdout);
50     #endif
51     int i,j,k,l;
52     int x,y,z;
53 //    for(scanf("%d",&cass);cass;cass--)
54 //    for(scanf("%d",&cas),cass=1;cass<=cas;cass++)
55 //    while(~scanf("%s",s))
56     while(~scanf("%d",&n))
57     {
58         scanf("%d",&m);
59         mem(dp,127);mem(maxx,0);sum[0]=0;
60         for(i=1;i<=n;i++)
61         {
62             scanf("%d",&a[i]);
63             sum[i]=sum[i-1]+a[i];
64             dp[i][i]=maxx[i][i]=a[i];
65         }
66         if(1+m+m>=n)
67         {
68             printf("%d\n",sum[n]);
69             continue;
70         }
71         for(i=1;i<=n;i++)
72             for(j=i+1;j<=n;j++)
73                 maxx[i][j]=maxx[j][i]=max(maxx[i][j-1],a[j]);
74         for(l=1;l<n-m-m;l++)
75         {
76             for(i=m+2;i+l<=n-m;i++)
77             {
78                 j=i+l;
79                 dp[i][j]=min(dp[i+1][j]+maxx[i][j],dp[j-1][i]+((i-m-1)+(n-j-m))*maxx[i][j]);
80             }
81             for(i=n-m;i-l>=m+2;i--)
82             {
83                 j=i-l;
84                 dp[i][j]=min(dp[i-1][j]+maxx[i][j],dp[j+1][i]+((j-m-1)+(n-i-m))*maxx[i][j]);
85             }
86         }
87         x=max(dp[m+2][n-m],dp[n-m][m+2])+sum[n];
88         printf("%d\n",x);
89     }
90     return 0;
91 }
92 /*
93 //
94
95 //
96 */

时间: 2024-12-20 16:38:08

Vijos 1451 圆环取数 【区间DP】的相关文章

Vijos1451圆环取数[环形DP|区间DP]

背景 小K攒足了路费来到了教主所在的宫殿门前,但是当小K要进去的时候,却发现了要与教主守护者进行一个特殊的游戏,只有取到了最大值才能进去Orz教主…… 描述 守护者拿出被划分为n个格子的一个圆环,每个格子上都有一个正整数,并且定义两个格子的距离为两个格子之间的格子数的最小值.环的圆心处固定了一个指针,一开始指向了圆环上的某一个格子,你可以取下指针所指的那个格子里的数以及与这个格子距离不大于k的格子的数,取一个数的代价即这个数的值.指针是可以转动的,每次转动可以将指针由一个格子转向其相邻的格子,且

[LuoguP1005]矩阵取数游戏 (DP+高精度)

题面 传送门:https://www.luogu.org/problemnew/show/P1005 Solution 我们可以先考虑贪心 我们每一次都选左右两边尽可能小的数,方便大的放在后面 听起来挺OK的 实则并不OK 反例: 3 1 1 1 1 1 2 2 2 2 2 2 2 2 2 2 如果贪心的话,我们会优先把右边那一串2先选了,再去选3和1 但是正确答案显然是先把3和1选了,再去选那一串2 既然贪心不成,我们可以考虑一下DP 然后我们考虑这样一个状态: f[i][j][k] 表示第i

方格取数(DP)(NOIP2000)

不要多想,我不是无聊到刷NOIP2000的题目,只是老师用考试告诉我们,我们的DP很弱,so咱家来找道DP提做 还有,这个题貌似和矩阵取数那个没有什么关系 题目简述:给你一个N*N的方格(小的很,N只有10),你可以向右或向下走一步,并取走其中的数字,你要取两次,输出最后取数的和 首先,这是一个棋盘DP,我们从左上到右下DP就好了,我一开始打的是记录位置的方法,跑两次DP,然而坑了,那我们要在仔细分析一下问题,N很小,小到你能开到6维不会爆内存,所以大胆的去提高维度吧,在分析问题,其实我们要找的

1083 矩阵取数问题(DP)

1083 矩阵取数问题 基准时间限制:1 秒 空间限制:131072 KB 分值: 5 难度:1级算法题  收藏  关注 一个N*N矩阵中有不同的正整数,经过这个格子,就能获得相应价值的奖励,从左上走到右下,只能向下向右走,求能够获得的最大价值. 例如:3 * 3的方格. 1 3 3 2 1 3 2 2 1 能够获得的最大价值为:11. Input 第1行:N,N为矩阵的大小.(2 <= N <= 500) 第2 - N + 1行:每行N个数,中间用空格隔开,对应格子中奖励的价值.(1 <

HDU - 1565 方格取数(1) (DP)

Description 给你一个n*n的格子的棋盘,每个格子里面有一个非负数. 从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取的数所在的2个格子不能相邻,并且取出的数的和最大. Input 包括多个测试实例,每个测试实例包括一个整数n 和n*n个非负数(n<=20) Output 对于每个测试实例,输出可能取得的最大的和 Sample Input 3 75 15 21 75 15 28 34 70 5 Sample Output 188 思路:用dp[i][j]表示到i行时下

Codeforces 509F Progress Monitoring 给定dfs序求树的同构数 区间dp

题目链接:点击打开链接 ==说同构数有点不对..反正就是这个意思,对于某个点的所有儿子,先访问标号小的,再访问标号大的. dp[l][r]表示 区间[l,r] 构成一棵树的方法数. 对于一个区间[l, r] 构成一棵树,则点l一定是根,然后枚举2个区间相乘即可 dp[l][r] = dp[l+1][i] * dp[i+1][r] ( i = [l+1, r] ) 当然 a[i+1] > a[l+1] ,这样才会满足题目中的暴力代码. import java.io.PrintWriter; imp

方格取数(DP)

描述 设有N*N的方格图(N<=10,我们将其中的某些方格中填入正整数,而其他的方格中则放入数字0.如下图所示(见样例): 某人从图的左上角的A 点出发,可以向下行走,也可以向右走,直到到达右下角的B点.在走过的路上,他可以取走方格中的数(取走后的方格中将变为数字0). 此人从A点到B 点共走两次,试找出2条这样的路径,使得取得的数之和为最大. 输入 输入的第一行为一个整数N(表示N*N的方格图),接下来的每行有三个整数,前两个表示位置,第三个数为该位置上所放的数.一行单独的0表示输入结束. 输

【noi 2.6_8786】方格取数(DP+优化)

题意:N*N的方格图每格有一个数值,要求从左上角每步往右或往下走到右下角,问走2次的最大和. 解法:走一次的很好想,而走2次,不可误以为先找到最大和的路,再找剩下的最大和的路就是正解.而应该认清动态规划的实质,定义为最佳解的状态,因此要走的2次都要涵括. O(n^4)——f[i][j][k][l]表示分别走到(i,j)和(k,l)的最大和.每次从上一步分别走(下,下),(右,右),(右,下),(下,右)的状态推导就好了.f[i][j][k][l]=max(f[i-1][j][k-1][l],f[

NOIP 2008 传纸条 NOIP 2000 方块取数 多线程DP

思路都是一样,建立一个四维dp然后跑一发就完了 当然,也可以像我这么帅的人,降成三维再傻傻的跑一发啦啦啦~ #include<iostream> #include<stdio.h> #include<math.h> #include<string.h> #include<memory.h> using namespace std; #define maxn 205 int dp[maxn][maxn][maxn]; int map[maxn][m