题目描述
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;
}