4. 多重背包问题 I
https://www.acwing.com/problem/content/4/
有 N种物品和一个容量是 V的背包。
第 i 种物品最多有 si 件,每件体积是 vi,价值是 wi。
求解将哪些物品装入背包,可使物品体积总和不超过背包容量,且价值总和最大。
输出最大价值。
输入格式
第一行两个整数,N,V,用空格隔开,分别表示物品种数和背包容积。
接下来有 N行,每行三个整数 vi,wi,si,用空格隔开,分别表示第 i种物品的体积、价值和数量。
输出格式
输出一个整数,表示最大价值。
数据范围
0<N,V≤100
0<vi,wi,si≤100
输入样例
4 5
1 2 3
2 4 1
3 4 3
4 5 2
输出样例:
10
解答
类似有多个01背包
复杂度 O(n*v*s)
#include <iostream>
using namespace std;
const int N = 105;
int dp[N][N];
struct NODE{
int v,w,s;
}node[N];
int n,v;
int main(){
cin >>n>>v;
for(int i=1;i<=n;i++){
cin >> node[i].v >>node[i].w>> node[i].s;
for(int j= 0;j <=v;j++){
for(int k=0;j>=k*node[i].v&&k<=node[i].s;k++){
dp[i][j] = max(dp[i][j],dp[i-1][j-k*node[i].v]+k*node[i].w);
}
}
}
cout << dp[n][v]<<endl;
}
但是复杂度 O(n*v*s),比较偏大 我们想办法优化。
上面代码中k件物品需要循环k次,但是我们将其化解为二进制,实际上只需要循环log(k)次。
比如某物品的件数位7,上面代码的循环是7次
我们在二进制下 7=20+21+2^2 即 7=1+2+4
可以将7件物品的多重背包 拆解成 1*v 2*v 4*v三件物品的01背包。循环次数从k降成logk
注意是从小到大逐步递增分解,不是按背包件数s的二进制数字
比如16件物品二进制是10000(二进制),分解是 1+2+4+8+1
int idx = 1;
int v, w, s; cin >> v >> w >> s;
int k = 1;
while (k <= s) {
node[idx].v = k * v;
node[idx].w = k * w;
idx++;
s -= k;
k <<= 1;
}
if (s > 0) {
node[idx].v = s * v;
node[idx].w = s * w;
idx++;
}
分解后按照01背包处理。 复杂度O(n*v*logk)
代码如下
#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int N = 12010, M = 2010;
int n, m;
int v[N], w[N];
int dp[M];
int a, b, s;
int main()
{
cin >> n >> m;
int cnt = 0;
for (int i = 1; i <= n; i++) {
cin >> a >> b >> s;
int k = 1;
while (k <= s) {
cnt++;
v[cnt] = k * a;
w[cnt] = k * b;
s -= k;
k <<= 1;
}
if (s > 0) {
cnt++;
v[cnt] = a * s;
w[cnt] = b * s;
}
}
n = cnt;
for (int i = 1; i <= n; i++) {
for (int j = m; j >= v[i]; j--)
dp[j] = max(dp[j], dp[j - v[i]] + w[i]);
}
cout << dp[m] << endl;
return 0;
}
作 者: itdef
欢迎转帖 请保持文本完整并注明出处
技术博客 http://www.cnblogs.com/itdef/
B站算法视频题解
https://space.bilibili.com/18508846
qq 151435887
gitee https://gitee.com/def/
欢迎c c++ 算法爱好者 windows驱动爱好者 服务器程序员沟通交流
如果觉得不错,欢迎点赞,你的鼓励就是我的动力
欢迎转帖 请保持文本完整并注明出处
技术博客 http://www.cnblogs.com/itdef/
B站算法视频题解
https://space.bilibili.com/18508846
qq 151435887
gitee https://gitee.com/def/
欢迎c c++ 算法爱好者 windows驱动爱好者 服务器程序员沟通交流
如果觉得不错,欢迎点赞,你的鼓励就是我的动力


【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 使用C#创建一个MCP客户端
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· 按钮权限的设计及实现
2020-06-07 算法问题实战策略 WILDCARD 递归 动态规划
2019-06-07 acwing 110 防晒
2015-06-07 c++11多线程学习笔记之二 mutex使用
2015-06-07 c++11多线程学习笔记之一 thread基础使用
2014-06-07 redis学习笔记
2014-06-07 ip白名单 通过* ? 检测IP匹配 轻量级