来源:牛客网
文章目录
- 题目描述
- 题解:
- 代码:
题目描述
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;
}