分类
背包模型的本质,就是从 n 种物品中选择若干,放入容量为 m 的背包。按照每种物品的数量,背包问题可以分成以下三种基本类型:
01背包
:每种物品只有 1 件,可以选择 0 件,可以选择 1 件完全背包
:每种物品数量无限,可以选择 0 件,可以选择 1 件,可以选择 2 件…只要已选物品的总重量不超过背包容量多重背包
:每种物品数量有限,可以选择 0 件,可以选择 1 件,可以选择 2 件…只要不超过该种物品的数量,且已选物品的总重量不超过背包容量
完全背包
解题
- 约定用
f[i][j]
表示从前 i 种物品中选择若干,放入容量为 j 的背包所能够获得的最大总价值,分别用 w 和 v 表示第 i 种物品的重量和价值
- 若选择 0 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j 的背包,获得的最大总价值为
f[i - 1][j]
- 若选择 1 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j - w 的背包,获得的最大总价值为
f[i - 1][j - w] + v
- 若选择 2 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j - 2 * w 的背包,获得的最大总价值为
f[i - 1][j - 2 * w] + 2 * v
- 。。。。。。
- 若选择 3 件第 i 种物品,等价于先从前 i - 1 种物品中选择若干,放入容量为 j - 3 * w 的背包,获得的最大总价值为
f[i - 1][j - 3 * w] + 3 * v
- 用变量来表示,若选择 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 ++