题解 SP283 【NAPTIME - Naptime】
原文食用效果更佳
题目链接
先考虑如果只有一天,那么该怎么做。
设
边界条件:
答案:
然后,现在两天是连着的,唯一的区别就是第一个小时可以睡着。
于是,令
因为SPOJ的空间限制比较大,我就没用滚动数组了。
还有就是楼下说要赋初值负无穷,有点不理解,难道还会出现负数吗?
#include <cstdio>
inline int max(int a, int b){
return a > b ? a : b;
}
inline int min(int a, int b){
return a < b ? a : b;
}
const int MAXN = 4000;
int f[MAXN][MAXN][3], w[MAXN];
int n, m, T;
int main(){
scanf("%d", &T);
while(T--){
int ans = 0;
scanf("%d%d", &n, &m);
for(int i = 1; i <= n; ++i)
scanf("%d", &w[i]);
f[1][0][0] = f[1][1][1] = 0;
for(int i = 2; i <= n; ++i){
for(int j = 1; j <= min(i, m); ++j){
f[i][j][0] = max(f[i - 1][j][0], f[i - 1][j][1]);
if(j != 1) f[i][j][1] = max(f[i - 1][j - 1][0], f[i - 1][j - 1][1] + w[i]); //第一个小时不加体力
}
}
f[1][1][1] = w[1]; f[1][0][0] = 0; //边界
ans = max(f[n][m][1], f[n][m][0]);
for(int i = 2; i <= n; ++i)
for(int j = 1; j <= min(i, m); ++j){
f[i][j][0] = max(f[i - 1][j][0], f[i - 1][j][1]);
if(j != 1) f[i][j][1] = max(f[i - 1][j - 1][0], f[i - 1][j - 1][1] + w[i]);
}
ans = max(ans, f[n][m][1]);
printf("%d\n", ans);
}
return 0;
}