poj 1390 Blocks 区间DP

Description:

给你一堆方块,颜色相同可以消去,每次消去的价值为消去方块的个数的平方。单个方块可以消去。问你最后收获的最大价值为多少

 1 #include<iostream>
 2 #include<cstring>
 3 #include<cstdio>
 4 using namespace std;
 5 const int N = 220;
 6 int dp[N][N][N], g[N][N], col[N]; 
 7 //dp[i][j][k]表示i-j这一段消得只剩k个且k个块都相同时收获的最大价值,g[i][j]表示i-j能收获的最大价值
 8 int n, m;
 9 int main(){
10     int T, kase = 0;
11     scanf("%d", &T);
12     while(T--){
13         scanf("%d", &n);
14         for(int i = 1; i <= n; i++) scanf("%d", &col[i]);
15         memset(dp, -63, sizeof(dp)); //达不到的状态为负无穷大
16         memset(g, 0, sizeof(g));
17         for(int i = 1; i <= n; i++)
18           dp[i][i][1] = 0, dp[i][i][0] = 0, g[i][i] = 1;
19         for(int l = 2; l <= n; l++)
20           for(int i = 1, j; i + l - 1 <= n; i++){   //枚举区间
21               j = i + l - 1;
22               if(col[i] == col[j]){  //如果区间首尾相同 说明可能通过消去区间内的方块更新dp数组
23                   for(int k = 1; k <= l; k++){  //枚举可能的相同剩余方块数
24                       for(int t = i; t < j; t++)  //枚举区间的每个方块
25                         if(col[i] == col[t])   //如果发现有方块与区间首方块相同
26                             //那么i到t的方块可以消掉t+1-j的方块后 达到dp[i][t][k-1]的状态
27                             //把两边的两边产生的价值累加之后更新dp数组
28                           dp[i][j][k] = max(dp[i][j][k], dp[i][t][k - 1] + g[t + 1][j - 1]);
29                       //那么区间g[i][j]的价值可以表现为状态dp[i][j][k]的价值和消掉剩下的k个方块的价值
30                       g[i][j] = max(g[i][j], dp[i][j][k] + k * k);  
31                   }
32               }
33               for(int k = i; k < j; k++)  //重新刷一波更新g数组
34                 g[i][j] = max(g[i][j], g[i][k] + g[k + 1][j]);
35           }
36         printf("Case %d: %d\n", ++kase, g[1][n]);
37     }
38     return 0;
39 }
View Code

 

posted @ 2018-07-06 22:53  Ror_shach  阅读(163)  评论(0编辑  收藏  举报