hdu3033 分组背包变形

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;
}


你可能感兴趣的:(hdu3033 分组背包变形)