IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 细说背包问题 - 完全背包 -> 正文阅读

[数据结构与算法]细说背包问题 - 完全背包

分类

背包模型的本质,就是从 n 种物品中选择若干,放入容量为 m 的背包。按照每种物品的数量,背包问题可以分成以下三种基本类型:

  • 01背包:每种物品只有 1 件,可以选择 0 件,可以选择 1 件
  • 完全背包:每种物品数量无限,可以选择 0 件,可以选择 1 件,可以选择 2 件…只要已选物品的总重量不超过背包容量
  • 多重背包:每种物品数量有限,可以选择 0 件,可以选择 1 件,可以选择 2 件…只要不超过该种物品的数量,且已选物品的总重量不超过背包容量

完全背包

在这里插入图片描述

解题

  • 约定用f[i][j]表示从前 i 种物品中选择若干,放入容量为 j 的背包所能够获得的最大总价值,分别用 w 和 v 表示第 i 种物品的重量和价值
  1. 若选择 0 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j 的背包,获得的最大总价值为f[i - 1][j]
  2. 若选择 1 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j - w 的背包,获得的最大总价值为f[i - 1][j - w] + v
  3. 若选择 2 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j - 2 * w 的背包,获得的最大总价值为f[i - 1][j - 2 * w] + 2 * v
  4. 。。。。。。
  5. 若选择 3 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j - 3 * w 的背包,获得的最大总价值为f[i - 1][j - 3 * w] + 3 * v
  6. 用变量来表示,若选择 k 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j - k * w 的背包,获得的最大总价值为f[i - 1][j - k * w] + k * v,k的值满足0 <= k <= j / w
  • 我们通过打擂台,找到从前 i 种物品中选择若干,放入容量为 j 的背包所能够获得的最大总价值
  • 题目要求的,从前 n 种物品中选择若干,放入容量为 m 的背包,能够获得的最大总价值,即f[n][m]

代码如下:

#include <bits/stdc++.h>
using namespace std;

int m, n;
int f[39][209];

int main() {
	cin >> m >> n;
	
	for (int i = 1; i <= n; i ++) {
		int w, v;
		cin >> w >> v;
		for (int j = 1; j <= m; j ++) {
			for (int k = 0; k <= j / w; k ++) {
				f[i][j] = max(f[i][j], f[i - 1][j - k * w] + k * v);
			}
		}
	}	
	cout << "max=" << f[n][m];

	return 0;
}

优化时间复杂度

这个代码无论时间复杂度还是空间复杂度,都不是最优的。我们进一步观察状态转移方程。

  • 如果背包容量j >= w,我们考虑以下两种情况:
  1. 从前 i 种物品中选择若干,放入容量为 j 的背包,从下表左侧这一列中找出的最大值就是f[i][j]
  2. 从前 i 种物品中选择若干,放入容量为 j - w 的背包,从下表右侧这一列中找出的最大值就是f[i][j - w]
前 i 种物品 -> 容量为 j 的背包前 i 种物品 -> 容量为 j - w 的背包左右两列相差
f[i - 1][j](选0件)不适用
f[i - 1][j - 1 * w] + 1 * v(选1件)f[i - 1][j - 1 * w](选0件)v
f[i - 1][j - 2 * w] + 2 * v(选2件)f[i - 1][j - 2 * w] + 1 * v(选1件)v
f[i - 1][j - 3 * w] + 3 * v(选3件)f[i - 1][j - 3 * w] + 2 * v(选2件)v
。。。。。。。。。。。。
  • 对比这两列,我们得出 f[i][j] = max(f[i - 1][j], f[i][j - w] + v
  • 这个新的状态转移方程中不再出现变量 k,也就是说,我们可以用嵌套循环取代三重循环!
  • 如果背包容量j < w,则f[i][j] = f[i - 1][j]

代码如下:

#include <bits/stdc++.h>
using namespace std;

int m, n;
int f[39][209];

int main() {
	cin >> m >> n;
	
	for (int i = 1; i <= n; i ++) {
		int w, v;
		cin >> w >> v;
		for (int j = 1; j <= m; j ++) {
			if (j < w) f[i][j] = f[i - 1][j];
			else f[i][j] = max(f[i - 1][j], f[i][j - w] + v);
		}
	}	
	cout << "max=" << f[n][m];

	return 0;
}

优化空间复杂度

  • 上面的代码使用二维数组,f[i][j]表示从前 i 种物品中选择若干,放入容量为 j 的背包能够获得的最大总价值。
  • 和处理01背包的方法一样,我们可以将二维数组 f 压缩成一维数组
  • f[i][j] = max(f[i - 1][j], f[i][j - w] + v)变成f[j] = max(f[j],f[j - w] + v
  • f[i][j] = f[i - 1][j]变成f[j] = f[j],这条语句可以省略
  • 这样一来,完全背包和01背包的状态转移方程变得一模一样。整个代码的区别在于,01背包的第二重循环,是从大到小枚举背包容量,而完全背包的第二重循环,是从小到大枚举背包容量。
  • 我们从这个角度来理解,假设有一个容量为 j 的背包,它装载物品的最大价值是f[j]。如果增加一件重量为 w 的物品后不会超过容量 m,则容量为 j + w的背包的最大价值f[j + w] = f[j] + v;如果再增加一件重量为 w 的物品不会超过容量 m,则容量为 j + 2 * w 的背包的最大价值f[j + 2 * w] = f[j + w] + v,…。当我们从小到大枚举背包容量的时候,恰好对应多次选择同一种物品。
    f[j + w] = f[j] + v,选择 1 次
    f[j + 2 * w] = f[j + w] + v,选择 2 次
    f[j + 3 * w] = f[j + 2 * w] + v,选择 3 次
    f[j + 4 * w] = f[j + 3 * w] + v,选择 4 次
    。。。。。。

代码如下

#include <bits/stdc++.h>
using namespace std;

int m, n;
int f[209];

int main() {
	cin >> m >> n;
	
	for (int i = 1; i <= n; i ++) {
		int w, v;
		cin >> w >> v;
		for (int j = w; j <= m; j ++) {
			f[j] = max(f[j], f[j - w] + v);
		}
	}	
	cout << "max=" << f[m];

	return 0;
}
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-04-30 08:55:25  更:2022-04-30 08:55:48 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/4 17:54:26-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码