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