来源:牛客网

@[toc]

题目描述

xinjun是各类手游的狂热粉丝,因随手一氪、一氪上千而威震工大,现在他迷上了阴阳师。xinjun玩手游有一个习惯,就是经过层层计算制定出一套方案来使操作利益最大化(因此即使有扫荡券也不用,故称圣雄肝帝)。已知阴阳师有N个模式可以操作,模式i有ai种操作,但每种模式每日只能选用一种操作,可以不选。操作j能收益vj,但需要花费体力wj点。xinjun每日拥有体力M点,求他每日最多能得到多少收益。

输入描述:

第一行一个正整数T(T<=10),表示共有T组数据。

对于每组数据,第一行两个正整数N,M(1<=N,M<=1000)。

接下来N段数据,每段第一行一个正整数ai(1<=ai<=1000),第二行ai个正整数vj(1<=vj<=1000),第三行ai个正整数wj(1<=wj<=1000)。

每组数据ai之和不大于104。

输出描述:

对每组数据输出一行,即xinjun每日最多能得到多少收益。

示例1
输入

1
3 10
2
2 3
3 2
2
1 1
3 4
1
5
5

输出

9

题解:

01背包,但是独特点在于,N个模式,每个模式有ai种操作,一个模式只能用一次操作,也就是并非所有操作都参与最终受益,相同模式下只能选一个操作。
这怎么解决呢?
我们用二维数字来存受益与花费
v[i][j]表示第i个模式下第j种操作的受益
w[i][j]表示第i个模式下第j种操作的花费
那么递推方程就是
f[j] = max(f[j],f[j-w[i][k]]+v[i][k]);
三重for循环的含义
在当前容量下,枚举每一种操作,保留最佳情况
我原本以为复杂度会超,发现我想太多了,数据有点水。。
这种方法应该是算是比较简单明了

代码:

#include<bits/stdc++.h>

using namespace std;

const int maxn = 1005;
int n,m;
int f[maxn];
int c[maxn];
int w[maxn][maxn];
int v[maxn][maxn];
int t;

int main()
{
    cin >> t;
    while(t--){
        cin >> n >> m;
        for(int i = 1; i <= n; ++i)
        {
            cin >> c[i];
            for(int j = 1; j <= c[i]; ++j)
            {
                cin >> v[i][j];
            }
            for(int j = 1; j <= c[i]; ++j)
            {
                cin >> w[i][j];
            }
        }
        memset(f,0,sizeof(f));
        f[0] = 0;
        for(int i = 1; i <= n; ++i)//模式 
        {
            for(int j = m; j >= 0; --j)
            {
                for(int k = 1; k <= c[i]; ++k)//第i种操作 
                {
                    if(j>=w[i][k])
                        f[j] = max(f[j],f[j-w[i][k]]+v[i][k]);
                }
            }
        }

        cout << f[m] << endl;
    }
    return 0;
}

还有个方式:
参考
模式与模式之间都是独立的
每一个模式不是只能选一种吗?那我就挑选出最佳的操作,然后进入下一个模式,一直这样走

#include<iostream>
#include<cstring>
using namespace std;
const int N = 1010;
struct node{
    int f;
    int ne;
}dp[N];

int main (){
    int T;
    cin>>T;
    while(T--){
        int P,W;
        scanf("%d %d",&P,&W);
        while(P--){
            int n;
            scanf("%d",&n);
            int w[n],v[n];
            for(int i=0;i<n;i++){
                scanf("%d",&v[i]);
            }
            for(int i=0;i<n;i++){
                scanf("%d",&w[i]);
            }

            for(int i=0;i<n;i++){
                for(int j=W;j>=w[i];j--){
                    dp[j].ne=max(dp[j].ne,dp[j-w[i]].f+v[i]);

                }
            }
            for(int i=0;i<=W;i++){
                dp[i].f=dp[i].ne;
            }
        }

        cout<<dp[W].f<<endl;
        memset(dp,0,sizeof dp);
    }
    return 0;
}