题目地址:点击打开链接
思路:01背包,具体看代码,还有本博客算法精讲类里Tianyi Cui大神写的背包9讲
错误代码:
#include<stdio.h> #include<string.h> int value[1010],volumn[1010],dp[1010][1010]; int max(int a,int b) { return a > b ? a : b; } int main() { int t,n,v,i,j; scanf("%d",&t); while(t--) { memset(dp,0,sizeof(dp)); scanf("%d%d",&n,&v); for(i=0; i<n; i++) { scanf("%d",&value[i]); } for(i=0; i<n; i++) { scanf("%d",&volumn[i]); } for(i=0; i<n; i++) { for(j=0; j<=v; j++) { if(j >= volumn[i])//能放下就取放与不放的最大值 dp[i][j] = max(dp[i-1][j],dp[i-1][j-volumn[i]] + value[i]);//i=0时i-1成负数了 else dp[i][j] = dp[i-1][j];//放不下就只能是上一阶段的值 } } printf("%d\n",dp[n-1][v]);//注意是n-1因为是从0开始赋值的 } return 0; }
二维数组
AC代码:
#include<stdio.h> #include<string.h> int value[1010],volumn[1010],dp[1010][1010]; int max(int a,int b) { return a > b ? a : b; } int main() { int t,n,v,i,j; scanf("%d",&t); while(t--) { memset(dp,0,sizeof(dp)); scanf("%d%d",&n,&v); for(i=1; i<=n; i++) { scanf("%d",&value[i]); } for(i=1; i<=n; i++) { scanf("%d",&volumn[i]); } for(i=1; i<=n; i++) { for(j=0; j<=v; j++) { if(j >= volumn[i])//能放下就取放与不放的最大值 dp[i][j] = max(dp[i-1][j],dp[i-1][j-volumn[i]] + value[i]); else dp[i][j] = dp[i-1][j];//放不下就只能是上一阶段的值 } } printf("%d\n",dp[n][v]);//注意是n-1因为是从0开始赋值的 } return 0; }
AC代码:
#include<stdio.h> #include<string.h> int value[1010],volumn[1010],dp[1010]; int max(int a,int b) { return a > b ? a : b; } int main() { int t,n,v,i,j; scanf("%d",&t); while(t--) { memset(dp,0,sizeof(dp)); scanf("%d%d",&n,&v); for(i=0; i<n; i++) { scanf("%d",&value[i]); } for(i=0; i<n; i++) { scanf("%d",&volumn[i]); } for(i=0; i<n; i++) { for(j=v; j>=volumn[i]; j--)//这里逆序的原因:使用一维数组时,当第i次循环之前时,f[v]实际上就是f[i-1][v],那么怎么得到第二个子问题的值呢?事实上,如果在每次循环中我们以v=v…0的顺序推f[v]时,就能保证f[v-c[i]]存储的是f[i-1][v-c[i]]的状态。状态转移方程为 { dp[j] = max(dp[j],dp[j-volumn[i]] + value[i]); } } printf("%d\n",dp[v]); } return 0; }