http://acm.hdu.edu.cn/showproblem.php?pid=3033
借鉴了别人的思路:
设计状态dp[i][j]代表前i组容量为j的最大价值。由于一组里面有多个物品,所以状态转移可以是前一组少取一个,即dp[i-1][p-g[i][j].v]+g[i][j].w,也可以是当前组之前去过的少取一种,即dp[i][p-g[i][j].v]+g[i][j].w。
网上有些解题报告是错误的解法,这题dp初始化的时候要初始化为负无穷,因为这题要求的是上一组恰好达到的状态才能转移到这一组来,因为每一组至少得去一个。当然可以初始化为-1,每次判断一下是不是-1就行。
#include <iostream> #include<cstdio> #include<cstring> using namespace std; struct node { int v,w; }; node g[15][105]; int num[15]; int dp[15][10005]; int main() { int m,n,k; while(scanf("%d%d%d",&n,&m,&k)!=EOF) { memset(num,0,sizeof(num)); int i; for(i=1;i<=n;i++) { int j,a,b; scanf("%d%d%d",&j,&a,&b); g[j][num[j]].w=a; g[j][num[j]].v=b; num[j]++; } int j; memset(dp,-1,sizeof(dp)); memset(dp[0],0,sizeof(dp[0])); for(i=1;i<=k;i++) { for(j=0;j<num[i];j++) { int p; for(p=m;p>=g[i][j].w;p--) { if(dp[i][p-g[i][j].w]!=-1) dp[i][p]=max(dp[i][p-g[i][j].w]+g[i][j].v,dp[i][p]); if(dp[i-1][p-g[i][j].w]!=-1) dp[i][p]=max(dp[i][p],dp[i-1][p-g[i][j].w]+g[i][j].v); } } } if(dp[k][m]==-1) printf("Impossible\n"); else printf("%d\n",dp[k][m]); } return 0; }