题目考点:二维线性DP (初学者可以参考代码对样例模拟状态转移的过程,有助于今后理解状态转移)
题目大意:n种花m个花瓶,花瓶顺序以及插入花的顺序固定,每种花在每个花瓶中美观度不同(有负数),问讲n种花插在哪n个花瓶里美观度最大。
题目分析:对于第i种花,它能获得的最大美观度为:在合法区间内,上一种花所在位置后面的花瓶中的最大美观度的位置(其中合法区间是指从上一种花(上一种花用k维护)放的位置开始,到使得i后面的话有花瓶可以放的区间,比如根据题目样例,对于第2种花来说,它最后能放的位置为4,后面至少留1个花瓶的位置)。
因此转移方程:dp [ i ][ j ] = max (dp [ i ][ j ], dp [ i-1 ][ k ] + v [ i ][ j ]);
代码:#include<iostream> #include<algorithm> #include<cstring> using namespace std; const int N = 105, INF = 0x3f3f3f3f; int dp[N][N], v[N][N], id[N]; int n, m; int main() { scanf("%d%d", &n, &m); for(int i = 1; i <= n; i++) for(int j = 1; j <= m; j++) //dp数组初始化极小值 scanf("%d", &v[i][j]) , dp[i][j] = -INF; for(int i = 1; i <= n; i++) for(int j = i; j <= m - (n-i); j++) //第i种花的合法区间(合法区间的定义写在上面了) for(int k = i-1; k < j; k++) //k维护上一层的摆放情况 dp[i][j] = max(dp[i][j], dp[i-1][k] + v[i][j]); //寻找最大美观度 int ans = -INF; for(int i = 1; i <= m; i++) ans = max(ans, dp[n][i]); printf("%d\n", ans); //寻找位置 int pos = n; for(int i = n; i ; i--) //倒推寻找位置 for(int j = 1; j <= m; j++) //对于每一种花,遍历所在层寻找 if(dp[i][j] == ans) //ans对应dp维护的值,证明找到了位置 { id[pos--] = j; ans -= v[i][j]; break; //该层位置找到后记得直接跳到再上一层寻找上一层的位置 } for(int i = 1; i <= n; i++) //输出n种花的位置 printf("%d ", id[i]); return 0; }