1798: 【提高】多重背包(2)
内存限制:128 MB
时间限制:1.000 S
评测方式:文本比较
命题人:
提交:3
解决:1
题目描述
有N种物品和一个容量是V的背包。
第i种物品最多有si件,每件体积是vi,价值是wi。
求解将哪些物品装入背包,可使物品体积总和不超过背包容量,且价值总和最大。
输出最大价值。
第i种物品最多有si件,每件体积是vi,价值是wi。
求解将哪些物品装入背包,可使物品体积总和不超过背包容量,且价值总和最大。
输出最大价值。
输入
第一行两个整数,N,V,用空格隔开,分别表示物品种数和背包容积。
接下来有 N 行,每行三个整数 vi,wi,si,用空格隔开,分别表示第 i 种物品的体积、价值和数量。
输出
输出一个整数,表示最大价值。
0<N≤1000
0<V≤2000
0<vi,wi,si≤2000
样例输入 复制
4 5
1 2 3
2 4 1
3 4 3
4 5 2
样例输出 复制
10
提示
#include <bits/stdc++.h>
using namespace std;
const int N = 20010;
int v[N],w[N],dp[2010];
int n,m; //n种物品,背包容量为 m
int vi,wi,si;
int k =0;
int main(){
cin>>n>>m;
for(int i= 1;i <= n;i++){
cin>>vi>>wi>>si;
/*对si二进制化,比如:有10件一样的物品
我们转换为有4件不同的物品:1 2 4 3
这4种物品的体积分别是:1*vi 2*vi 4*vi 3*vi
*/
int t = 1; //权重,表示2的次方
while(t <= si){
k++;
v[k]= t*vi;
w[k]= t*wi;
si=si-t;
t=t*2;
}
//如果二进制化有剩余,存入
if(si > 0){
k++;
v[k] = si *vi;
w[k] = si *wi;
}
}
//01背包
for(int i = 1;i <= k;i++){
for(int j= m;j >= v[i];j--){
dp[j] = max(dp[j],dp[j-v[i]]+w[i]);
}
}
cout<<dp[m];
return 0;
}