动态规划专题——背包问题

🧑‍💻 文章作者:Iareges
🔗 博客主页:https://blog.csdn.net/raelum
⚠️ 转载请注明出处

目录

  • 前言
  • 一、01背包
    • 1.1 使用滚动数组优化
  • 二、完全背包
    • 2.1 使用滚动数组优化
  • 三、多重背包
    • 3.1 使用二进制优化
  • 四、分组背包
  • 总结

前言

本文主要介绍常见的四种背包问题,思维导图如下:

一、01背包

💡 现有 N N N 件物品和一个最多能承重 M M M 的背包,第 i i i 件物品的重量是 w i w_i wi,价值是 v i v_i vi。在背包能承受的范围内,试问将哪些物品装入背包后可使总价值最大,求这个最大价值。

因为每件物品只有选与不选两种状态,所以该问题又称01背包问题。

d p [ i ] [ j ] dp[i][j] dp[i][j] 的含义是:在背包承重为 j j j 的前提下,从前 i i i 个物品中选能够得到的最大价值。不难发现 d p [ N ] [ M ] dp[N][M] dp[N][M] 就是本题的答案。

如何计算 d p [ i ] [ j ] dp[i][j] dp[i][j] 呢?我们可以将它划分为以下两部分:

  • 选第 i i i 个物品:由于第 i i i 个物品一定会被选择,那么相当于从前 i − 1 i-1 i1 个物品中选且总重量不超过 j − w [ i ] j-w[i] jw[i],对应 d p [ i − 1 ] [ j − w [ i ] ] + v [ i ] dp[i-1][j-w[i]]+v[i] dp[i1][jw[i]]+v[i]
  • 不选第 i i i 个物品:意味着从前 i − 1 i-1 i1 个物品中选且总重量不超过 j j j,对应 d p [ i − 1 ] [ j ] dp[i-1][j] dp[i1][j]

结合以上两点可得递推公式:

d p [ i ] [ j ] = max ⁡ ( d p [ i − 1 ] [ j ] , d p [ i − 1 ] [ j − w [ i ] ] + v [ i ] ) dp[i][j] = \max(dp[i-1][j],\;dp[i-1][j-w[i]]+v[i]) dp[i][j]=max(dp[i1][j],dp[i1][jw[i]]+v[i])

由于下标不能是负数,所以上述递推公式要求 j ≥ w [ i ] j\geq w[i] jw[i]。当 j < w [ i ] j<w[i] j<w[i] 时,意味着第 i i i 个物品无法装进背包,此时 d p [ i ] [ j ] = d p [ i − 1 ] [ j ] dp[i][j]=dp[i-1][j] dp[i][j]=dp[i1][j]。综合以上可得出:

d p [ i ] [ j ] = { d p [ i − 1 ] [ j ] , j < w [ i ] max ⁡ ( d p [ i − 1 ] [ j ] , d p [ i − 1 ] [ j − w [ i ] ] + v [ i ] ) , j ≥ w [ i ] dp[i][j]= \begin{cases} dp[i-1][j],&j<w[i] \\ \max(dp[i-1][j],\;dp[i-1][j-w[i]]+v[i]),&j\geq w[i] \end{cases} dp[i][j]={dp[i1][j],max(dp[i1][j],dp[i1][jw[i]]+v[i]),j<w[i]jw[i]

d p dp dp 数组应当如何初始化呢?当背包承重为 0 0 0 时,显然装不下任何物品,所以 d p [ i ] [ 0 ] = 0 ( 1 ≤ i ≤ N ) dp[i][0]=0\;(1\leq i\leq N) dp[i][0]=0(1iN)。若一个物品也不选(即从前 0 0 0 个物品中选),此时最大价值也是 0 0 0,所以 d p [ 0 ] [ j ] = 0 ( 0 ≤ j ≤ M ) dp[0][j]=0\;(0\leq j\leq M) dp[0][j]=0(0jM)。由此可知, d p dp dp 数组应当全0初始化,即声明为全局变量。

题目链接:AcWing 2. 01背包问题

#include <bits/stdc++.h>using namespace std;const int N = 1010;int w[N], v[N];
int dp[N][N];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m;cin >> n >> m;for (int i = 1; i <= n; i++) cin >> w[i] >> v[i];for (int i = 1; i <= n; i++) {for (int j = 1; j <= m; j++) {if (j < w[i]) dp[i][j] = dp[i - 1][j];else dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - w[i]] + v[i]);}}cout << dp[n][m] << "\n";return 0;
}

时间复杂度为 O ( n m ) O(nm) O(nm)

1.1 使用滚动数组优化

之前我们用到的 d p dp dp 数组是二维数组,它可以进一步优化成一维数组。

观察递推公式不难发现, d p dp dp 数组中第 i i i 行的元素仅由第 i − 1 i-1 i1 行的元素得来,即第 0 0 0 行元素的更新值放到第 1 1 1 行,第 1 1 1 行元素的更新值放到第 2 2 2 行,以此类推。与其把一行的更新值放到新的一行,不如直接就地更新,因此我们的 d p dp dp 数组只需要一行来存储,即一维数组。

去掉 d p dp dp 数组的第一维后,递推公式变成:

d p [ j ] = { d p [ j ] , j < w [ i ] max ⁡ ( d p [ j ] , d p [ j − w [ i ] ] + v [ i ] ) , j ≥ w [ i ] dp[j]= \begin{cases} dp[j],&j<w[i] \\ \max(dp[j],\;dp[j-w[i]]+v[i]),&j\geq w[i] \end{cases} dp[j]={dp[j],max(dp[j],dp[jw[i]]+v[i]),j<w[i]jw[i]

d p [ j ] = max ⁡ ( d p [ j ] , d p [ j − w [ i ] ] + v [ i ] ) , j ≥ w [ i ] dp[j]= \max(dp[j],\;dp[j-w[i]]+v[i]),\quad j\geq w[i] dp[j]=max(dp[j],dp[jw[i]]+v[i]),jw[i]

原先 j j j 是从 1 1 1 遍历至 m m m 的,现在只需从 w [ i ] w[i] w[i] 遍历至 m m m。但,这个遍历顺序真的对吗?

请看下图:

红色箭头表示,在二维数组中, d p [ i ] [ j ] dp[i][j] dp[i][j] d p [ i − 1 ] [ j − w [ i ] ] dp[i-1][j-w[i]] dp[i1][jw[i]] d p [ i − 1 ] [ j ] dp[i-1][j] dp[i1][j] 得来, d p [ i ] [ j + w [ i ] ] dp[i][j+w[i]] dp[i][j+w[i]] d p [ i − 1 ] [ j ] dp[i-1][j] dp[i1][j] d p [ i − 1 ] [ j + w [ i ] ] dp[i-1][j+w[i]] dp[i1][j+w[i]] 得来。用一维数组的话来讲就是,第 i i i 行的 d p [ j ] dp[j] dp[j] 由第 i − 1 i-1 i1 行的 d p [ j − w [ i ] ] dp[j-w[i]] dp[jw[i]] d p [ j ] dp[j] dp[j] 得来,第 i i i 行的 d p [ j + w [ i ] ] dp[j+w[i]] dp[j+w[i]] 由第 i − 1 i-1 i1 行的 d p [ j ] dp[j] dp[j] d p [ j + w [ i ] ] dp[j+w[i]] dp[j+w[i]] 得来。

如果 j j j 从小到大遍历,那么会先更新 d p [ j ] dp[j] dp[j] 再更新 d p [ j + w [ i ] ] dp[j+w[i]] dp[j+w[i]],这就导致在更新 d p [ j + w [ i ] ] dp[j+w[i]] dp[j+w[i]] 时使用的是第 i i i 行的 d p [ j ] dp[j] dp[j] 而非第 i − 1 i-1 i1 行的 d p [ j ] dp[j] dp[j],即当 j j j 从小到大遍历时,二维数组的递推式变成了:

d p [ i ] [ j ] = { d p [ i − 1 ] [ j ] , j < w [ i ] max ⁡ ( d p [ i − 1 ] [ j ] , d p [ i ] [ j − w [ i ] ] + v [ i ] ) , j ≥ w [ i ] dp[i][j]= \begin{cases} dp[i-1][j],&j<w[i] \\ \max(dp[i-1][j],\;dp[i][j-w[i]]+v[i]),&j\geq w[i] \end{cases} dp[i][j]={dp[i1][j],max(dp[i1][j],dp[i][jw[i]]+v[i]),j<w[i]jw[i]

⚠️ 请牢记该式,后续讲解完全背包时会提到它。

这显然是错误的。事实上,让 j j j 从大到小遍历就不会出现这个问题。

#include <bits/stdc++.h>using namespace std;const int N = 1010;int w[N], v[N];
int dp[N];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m;cin >> n >> m;for (int i = 1; i <= n; i++) cin >> w[i] >> v[i];for (int i = 1; i <= n; i++)for (int j = m; j >= w[i]; j--)dp[j] = max(dp[j], dp[j - w[i]] + v[i]);cout << dp[m] << "\n";return 0;
}

当然, w w w 数组和 v v v 数组也是不必要的,我们可以边输入边处理,因此可以得到01背包问题的最终版代码:

#include <bits/stdc++.h>using namespace std;const int N = 1010;int dp[N];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m;cin >> n >> m;for (int i = 1; i <= n; i++) {int w, v;cin >> w >> v;for (int j = m; j >= w; j--)dp[j] = max(dp[j], dp[j - w] + v);}cout << dp[m] << "\n";return 0;
}

到此为止,可以总结出,当 d p dp dp 数组是二维数组时, j j j 既可以从小到大遍历也可以从大到小遍历,但当 d p dp dp 数组是一维数组时, j j j 只能从大到小遍历

二、完全背包

💡 现有 N N N 种物品和一个最多能承重 M M M 的背包,每种物品都有无限个,第 i i i 种物品的重量是 w i w_i wi,价值是 v i v_i vi。在背包能承受的范围内,试问将哪些物品装入背包后可使总价值最大,求这个最大价值。

d p [ i ] [ j ] dp[i][j] dp[i][j] 的含义是:在背包承重为 j j j 的前提下,从前 i i i 物品中选能够得到的最大价值。

如何计算 d p [ i ] [ j ] dp[i][j] dp[i][j] 呢?我们可以将它划分为以下若干部分:

  • 0 0 0 个第 i i i 种物品:相当于不选第 i i i 种物品,对应 d p [ i − 1 ] [ j ] dp[i-1][j] dp[i1][j]
  • 1 1 1 个第 i i i 种物品:对应 d p [ i − 1 ] [ j − w [ i ] ] + v [ i ] dp[i-1][j-w[i]]+v[i] dp[i1][jw[i]]+v[i]
  • 2 2 2 个第 i i i 种物品:对应 d p [ i − 1 ] [ j − 2 ⋅ w [ i ] ] + 2 ⋅ v [ i ] dp[i-1][j-2\cdot w[i]]+2\cdot v[i] dp[i1][j2w[i]]+2v[i]
  • ⋯ \cdots

上述过程并不会无限进行下去,因为背包承重是有限的。设第 i i i 种物品最多能选 t t t 个,于是可知 t = ⌊ j w [ i ] ⌋ t=\lfloor \frac{j}{w[i]}\rfloor t=w[i]j,从而得到递推式:

d p [ i ] [ j ] = max ⁡ 0 ≤ k ≤ t d p [ i − 1 ] [ j − k ⋅ w [ i ] ] + k ⋅ v [ i ] dp[i][j]=\max_{0\leq k\leq t} dp[i-1][j-k\cdot w[i]]+k\cdot v[i] dp[i][j]=0ktmaxdp[i1][jkw[i]]+kv[i]

题目链接:AcWing 3. 完全背包问题

#include <bits/stdc++.h>using namespace std;const int N = 1010;int w[N], v[N];
int dp[N][N];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m;cin >> n >> m;for (int i = 1; i <= n; i++) cin >> w[i] >> v[i];for (int i = 1; i <= n; i++)for (int j = 1; j <= m; j++) {int t = j / w[i];for (int k = 0; k <= t; k++)dp[i][j] = max(dp[i][j], dp[i - 1][j - k * w[i]] + k * v[i]);}cout << dp[n][m] << "\n";return 0;
}

若将 t t t 的值改为 min ⁡ ( 1 , j / w [ i ] ) \min(1,\,j/w[i]) min(1,j/w[i]),则完全背包将退化为01背包。

上述代码的时间复杂度为 O ( m 2 ∑ i w i − 1 ) ≈ O ( m 2 n ) O(m^2\sum_iw_i^{-1})\approx O(m^2n) O(m2iwi1)O(m2n),TLE是必然的。

2.1 使用滚动数组优化

考虑 d p [ i ] [ j ] dp[i][j] dp[i][j],此时第 i i i 种物品最多能选 t 1 = ⌊ j w [ i ] ⌋ t_1=\lfloor \frac{j}{w[i]} \rfloor t1=w[i]j 个,将递推式展开:

d p [ i ] [ j ] = max ⁡ ( d p [ i − 1 ] [ j ] , d p [ i − 1 ] [ j − w [ i ] ] + v [ i ] , d p [ i − 1 ] [ j − 2 ⋅ w [ i ] ] + 2 ⋅ v [ i ] , ⋮ d p [ i − 1 ] [ j − t 1 ⋅ w [ i ] ] + t 1 ⋅ v [ i ] ) \begin{aligned} dp[i][j] = \max(dp[i-1][j],\; &dp[i-1][j-w[i]]+v[i], \\ &dp[i-1][j-2\cdot w[i]]+2\cdot v[i], \\ &\vdots \\ &dp[i-1][j-t_1\cdot w[i]]+t_1\cdot v[i]) \end{aligned} dp[i][j]=max(dp[i1][j],dp[i1][jw[i]]+v[i],dp[i1][j2w[i]]+2v[i],dp[i1][jt1w[i]]+t1v[i])

下面考虑 d p [ i ] [ j − w [ i ] ] dp[i][j-w[i]] dp[i][jw[i]],此时第 i i i 种物品最多能选 t 2 = ⌊ j − w [ i ] w [ i ] ⌋ = ⌊ j w [ i ] − 1 ⌋ = t 1 − 1 t_2=\lfloor \frac{j-w[i]}{w[i]} \rfloor=\lfloor \frac{j}{w[i]}-1\rfloor=t_1-1 t2=w[i]jw[i]=w[i]j1=t11 个,相应的递推式为

d p [ i ] [ j − w [ i ] ] = max ⁡ ( d p [ i − 1 ] [ j − w [ i ] ] , d p [ i − 1 ] [ j − w [ i ] − w [ i ] ] + v [ i ] , d p [ i − 1 ] [ j − w [ i ] − 2 ⋅ w [ i ] ] + 2 ⋅ v [ i ] , ⋮ d p [ i − 1 ] [ j − w [ i ] − t 2 ⋅ w [ i ] ] + t 2 ⋅ v [ i ] ) \begin{aligned} dp[i][j-w[i]] = \max(dp[i-1][j-w[i]],\; &dp[i-1][j-w[i]-w[i]]+v[i], \\ &dp[i-1][j-w[i]-2\cdot w[i]]+2\cdot v[i], \\ &\vdots \\ &dp[i-1][j-w[i]-t_2\cdot w[i]]+t_2\cdot v[i]) \end{aligned} dp[i][jw[i]]=max(dp[i1][jw[i]],dp[i1][jw[i]w[i]]+v[i],dp[i1][jw[i]2w[i]]+2v[i],dp[i1][jw[i]t2w[i]]+t2v[i])

又注意到 t 1 = t 2 + 1 t_1=t_2+1 t1=t2+1,上式可化简为

d p [ i ] [ j − w [ i ] ] = max ⁡ ( d p [ i − 1 ] [ j − w [ i ] ] , d p [ i − 1 ] [ j − 2 ⋅ w [ i ] ] + v [ i ] , d p [ i − 1 ] [ j − 3 ⋅ w [ i ] ] + 2 ⋅ v [ i ] , ⋮ d p [ i − 1 ] [ j − t 1 ⋅ w [ i ] ] + ( t 1 − 1 ) ⋅ v [ i ] ) \begin{aligned} dp[i][j-w[i]] = \max(dp[i-1][j-w[i]],\; &dp[i-1][j-2\cdot w[i]]+v[i], \\ &dp[i-1][j-3\cdot w[i]]+2\cdot v[i], \\ &\vdots \\ &dp[i-1][j-t_1\cdot w[i]]+(t_1-1)\cdot v[i]) \end{aligned} dp[i][jw[i]]=max(dp[i1][jw[i]],dp[i1][j2w[i]]+v[i],dp[i1][j3w[i]]+2v[i],dp[i1][jt1w[i]]+(t11)v[i])

将该式与 d p [ i ] [ j ] dp[i][j] dp[i][j] 的递推式比较不难发现

d p [ i ] [ j ] = max ⁡ ( d p [ i − 1 ] [ j ] , d p [ i ] [ j − w [ i ] ] + v [ i ] ) dp[i][j]=\max(dp[i-1][j],\;dp[i][j-w[i]]+v[i]) dp[i][j]=max(dp[i1][j],dp[i][jw[i]]+v[i])

根据1.1节中的结论,该式对应的是 j j j 从小到大遍历,于是我们只需把01背包问题的代码中的 j j j 改为从小到大遍历即可

#include <bits/stdc++.h>using namespace std;const int N = 1010;int dp[N];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m;cin >> n >> m;for (int i = 1; i <= n; i++) {int w, v;cin >> w >> v;for (int j = w; j <= m; j++)  // 只需修改这一行dp[j] = max(dp[j], dp[j - w] + v);}cout << dp[m] << "\n";return 0;
}

优化后的时间复杂度为 O ( n m ) O(nm) O(nm)

三、多重背包

💡 现有 N N N 种物品和一个最多能承重 M M M 的背包,第 i i i 种物品的数量是 s i s_i si,重量是 w i w_i wi,价值是 v i v_i vi。在背包能承受的范围内,试问将哪些物品装入背包后可使总价值最大,求这个最大价值。

回顾完全背包问题的暴力解法,在背包承重为 j j j 的前提下,第 i i i 种物品最多能放 t = j / w [ i ] t=j/w[i] t=j/w[i] 个(这里是整除)。而在01背包问题中,第 i i i 种物品只有一个,所以应当取 t = min ⁡ ( 1 , j / w [ i ] ) t=\min(1,\,j/w[i]) t=min(1,j/w[i])。由此可见,对于多重背包问题,只需取 t = min ⁡ ( s [ i ] , j / w [ i ] ) t=\min(s[i],\,j/w[i]) t=min(s[i],j/w[i])

对完全背包问题的暴力解法做一点简单修改即可求解多重背包问题。

题目链接:AcWing 4. 多重背包问题

#include <bits/stdc++.h>using namespace std;const int N = 110;int dp[N][N];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m;cin >> n >> m;for (int i = 1; i <= n; i++) {int w, v, s;cin >> w >> v >> s;for (int j = 1; j <= m; j++) {int t = min(s, j / w);  // 只有这里需要修改for (int k = 0; k <= t; k++)dp[i][j] = max(dp[i][j], dp[i - 1][j - k * w] + k * v);}}cout << dp[n][m] << "\n";return 0;
}

时间复杂度为 O ( m ∑ i s i ) O(m\sum_i s_i) O(misi),但还可以进一步优化。

3.1 使用二进制优化

从时间复杂度的表达式可以看出, O ( m ) O(m) O(m) 的部分已经无法再优化了,我们只能从 O ( ∑ i s i ) O(\sum_i s_i) O(isi) 入手。

先来看一个例子。水果店里有 40 40 40 个苹果,小明计划购买 n ( 1 ≤ n ≤ 40 ) n\,(1\leq n\leq 40) n(1n40) 个苹果,试问如何让小明尽可能快速地完成购买?一个显而易见的暴力做法是,让小明一个个拿(单位是个),但效率过于低下。事实上,店员可事先准备好 6 6 6 个箱子,每个箱子中的苹果数量分别为 [ 1 , 2 , 4 , 8 , 16 , 9 ] [1,2,4,8,16,9] [1,2,4,8,16,9],再让小明按箱子拿(单位是箱子),无论小明计划购买多少个,他最多只需要拿 6 6 6 次,而在暴力做法中,小明最多需要拿 40 40 40 次。

下面用数学语言来描述上面的例子。对于任意的正整数 s s s,我们都可以找到 ⌊ log ⁡ 2 s ⌋ + 1 ≜ k \lfloor \log_2 s\rfloor+1\triangleq k log2s+1k 个正整数 a 1 , ⋯ , a k a_1,\cdots,a_k a1,,ak,使得 ∀ n ∈ [ 0 , s ] \forall\, n\in[0,s] n[0,s],都有

n = v T a , a = ( a 1 , ⋯ , a k ) T , a i = { 2 i − 1 , 1 ≤ i ≤ k − 1 s − 2 k − 1 + 1 ( ∈ [ 1 , 2 k − 1 ] ) , i = k n=v^\mathrm{T}a,\quad a=(a_1,\cdots,a_k)^\mathrm{T},\quad a_i= \begin{cases} 2^{i-1},&1\leq i\leq k -1\\ s-2^{k-1}+1\,(\in [1,\,2^{k-1}]),&i=k\\ \end{cases} n=vTa,a=(a1,,ak)T,ai={2i1,s2k1+1([1,2k1]),1ik1i=k

其中 v = ( v 1 , ⋯ , v k ) T v=(v_1,\cdots,v_k)^\mathrm{T} v=(v1,,vk)T,且其分量非 0 0 0 1 1 1

感兴趣的读者可自行证明,这里不再赘述。回到本题,先不考虑背包的承重,我们在暴力求解多重背包的时候,对于每种物品 i i i,都要从 0 0 0 逐个枚举至 s [ i ] s[i] s[i],效率无疑是低下的。现在,对于每种物品 i i i,我们将这 s [ i ] s[i] s[i] 个物品分散至 ⌊ log ⁡ 2 s [ i ] ⌋ + 1 \lfloor \log_2 s[i]\rfloor+1 log2s[i]⌋+1 个「箱子」中,于是多重背包便化成了01背包。

题目链接:AcWing 5. 多重背包问题 II

多重背包问题中的一个「箱子」相当于01背包问题中的一件「物品」,因此我们需要估计出多重背包问题中到底有多少个箱子。显然箱子总数为

N = ∑ i = 1 n ( ⌊ log ⁡ 2 s [ i ] ⌋ + 1 ) ≤ ∑ i = 1 n ⌊ log ⁡ 2 2000 ⌋ + n = 11 n ≤ 11000 N=\sum_{i=1}^n(\lfloor \log_2 s[i]\rfloor+1)\leq \sum_{i=1}^n \lfloor \log_2 2000\rfloor+n=11n\leq 11000 N=i=1n(⌊log2s[i]⌋+1)i=1nlog22000+n=11n11000

#include <bits/stdc++.h>using namespace std;const int N = 11010, M = 2010;int w[N], v[N];
int dp[M];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m;cin >> n >> m;int cnt = 0;while (n--) {int a, b, s;  // a是重量, b是价值, c是数量cin >> a >> b >> s;for (int k = 1; k <= s; k *= 2) {cnt++;w[cnt] = a * k, v[cnt] = b * k;s -= k;}if (s > 0) {cnt++;w[cnt] = a * s, v[cnt] = b * s;}}n = cnt;for (int i = 1; i <= n; i++)for (int j = m; j >= w[i]; j--)dp[j] = max(dp[j], dp[j - w[i]] + v[i]);cout << dp[m] << "\n";return 0;
}

优化后的时间复杂度为 O ( m ∑ i log ⁡ s i ) O(m\sum_i \log s_i) O(milogsi)

四、分组背包

💡 现有 N N N 组物品和一个最多能承重 M M M 的背包,每组物品有若干个,同一组内的物品最多只能选一个。每件物品的重量是 w i j w_{ij} wij,价值是 v i j v_{ij} vij,其中 i i i 是组号, j j j 是组内编号。在背包能承受的范围内,试问将哪些物品装入背包后可使总价值最大,求这个最大价值。

d p [ i ] [ j ] dp[i][j] dp[i][j] 的含义是:在背包承重为 j j j 的前提下,从前 i i i 物品中选能够得到的最大价值。

如何计算 d p [ i ] [ j ] dp[i][j] dp[i][j] 呢?我们可以将它划分为以下若干部分:

  • 不选第 i i i 组的物品:对应 d p [ i − 1 ] [ j ] dp[i-1][j] dp[i1][j]
  • 选第 i i i 组的第 1 1 1 个物品:对应 d p [ i − 1 ] [ j − w [ i ] [ 1 ] ] + v [ i ] [ 1 ] dp[i-1][j-w[i][1]]+v[i][1] dp[i1][jw[i][1]]+v[i][1]
  • 选第 i i i 组的第 2 2 2 个物品:对应 d p [ i − 1 ] [ j − w [ i ] [ 2 ] ] + v [ i ] [ 2 ] dp[i-1][j-w[i][2]]+v[i][2] dp[i1][jw[i][2]]+v[i][2]
  • ⋯ \cdots
  • 选第 i i i 组的第 s [ i ] s[i] s[i] 个物品:对应 d p [ i − 1 ] [ j − w [ i ] [ s [ i ] ] ] + v [ i ] [ s [ i ] ] dp[i-1][j-w[i][s[i]]]+v[i][s[i]] dp[i1][jw[i][s[i]]]+v[i][s[i]]

直接将 d p dp dp 数组优化到一维可得递推式:

d p [ j ] = max ⁡ ( d p [ j ] , max ⁡ 1 ≤ k ≤ s [ i ] d p [ j − w [ i ] [ k ] ] + v [ i ] [ k ] ) dp[j]=\max(dp[j],\;\max_{1\leq k\le s[i]} dp[j-w[i][k]]+v[i][k]) dp[j]=max(dp[j],1ks[i]maxdp[jw[i][k]]+v[i][k])

题目链接:AcWing 9. 分组背包问题

#include <bits/stdc++.h>using namespace std;const int N = 110;int w[N][N], v[N][N], s[N];
int dp[N];int main() {ios::sync_with_stdio(false);cin.tie(nullptr);int n, m;cin >> n >> m;for (int i = 1; i <= n; i++) {cin >> s[i];for (int j = 1; j <= s[i]; j++)cin >> w[i][j] >> v[i][j];}for (int i = 1; i <= n; i++)for (int j = m; j >= 1; j--)for (int k = 1; k <= s[i]; k++)if (j >= w[i][k])dp[j] = max(dp[j], dp[j - w[i][k]] + v[i][k]);cout << dp[m] << "\n";return 0;
}

总结

我们可以用一个公式来表示01背包、完全背包和多重背包:

d p [ i ] [ j ] = max ⁡ 0 ≤ k ≤ t d p [ i − 1 ] [ j − k ⋅ w [ i ] ] + k ⋅ v [ i ] , t = { min ⁡ ( 1 , j / w [ i ] ) , 01 背包 min ⁡ ( + ∞ , j / w [ i ] ) = j / w [ i ] , 完全背包 min ⁡ ( s [ i ] , j / w [ i ] ) , 多重背包 dp[i][j]=\max_{0\leq k\leq t} dp[i-1][j-k\cdot w[i]]+k\cdot v[i],\quad t=\begin{cases} \min(1,\;j/w[i]),&01背包\\ \min(+\infty,\;j/w[i])=j/w[i],&完全背包 \\ \min(s[i],\;j/w[i]),&多重背包 \end{cases} dp[i][j]=0ktmaxdp[i1][jkw[i]]+kv[i],t= min(1,j/w[i]),min(+,j/w[i])=j/w[i],min(s[i],j/w[i]),01背包完全背包多重背包

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/173542.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Codeforces Round 911 (Div. 2) --- D题题解

D. Small GCD Problem - D - Codeforces 题目大意&#xff1a; 给你一个数组&#xff0c;你可以在里面任选三个数ai aj ak&#xff0c;要求i j k 互不相同&#xff0c; 现定义一个函数f(a,b,c)gcd(a,b)&#xff0c;其中a 和 b为a&#xff0c;b&#xff0c;c中较小的两个。求f…

大数据平台/大数据技术与原理-实验报告--MapReduce编程

实验名称 MapReduce编程 实验性质 &#xff08;必修、选修&#xff09; 必修 实验类型&#xff08;验证、设计、创新、综合&#xff09; 综合 实验课时 2 实验日期 2023.10.30-2023.11.03 实验仪器设备以及实验软硬件要求 专业实验室&#xff08;配有centos7.5系统…

【代码随想录刷题】Day18 二叉树05

文章目录 1.【513】找树左下角的值1.1题目描述1.2 解题思路1.2.1 迭代法思路1.2.2 递归法思路 1.3 java代码实现1.3.1 迭代法java代码实现1.3.2 递归法java代码实现 2. 【112】路径总和2.1题目描述2.2 解题思路2.3 java代码实现 3.【106】从中序与后序遍历序列构造二叉树3.1题目…

Linux | Linux入门及常用基础命令介绍

关注CodingTechWork Linux Linux介绍 概述 Linux出现的时候是没有图像化界面&#xff0c;都是黑屏操作&#xff0c;靠命令来完成操作&#xff0c;如磁盘读写、网络管理等。企业级服务器的维护基本都通过跳板机ssh到对应的服务器上进行操作&#xff0c;一般无图形化界面。 远…

DevEco Studio对同一套HarmonyOS代码进行多设备端预览

鸿蒙代码有一个很大的优势 不需要其他的语法 只需要一套HarmonyOS代码 就可以在 手机 平板 电脑上运行 我们可以在DevEco Studio预览器上 点击如下图指向位置 弹出的这个窗口中 我们将右上角的开关勾选上 这样 我们调试器向下滚动 就可以看到多端预览的一个效果了

Windows Server 2012R2 修复CVE-2016-2183(SSL/TLS)漏洞的办法

一、漏洞说明 Windows server 2012R2远程桌面服务SSL加密默认是开启的,且有默认的CA证书。由于SSL/ TLS自身存在漏洞缺陷,当开启远程桌面服务,使用漏洞扫描工具扫描,发现存在SSL/TSL漏洞。远程主机支持的SSL加密算法提供了中等强度的加密算法,目前,使用密钥长度大于等于5…

队列实现方式、效率分析及应用场景

文章目录 一、什么是队列二、队列特性阻塞和非阻塞有界和无界单向链表和双向链表 三、Java队列接口继承图四、Java队列常用方法五、队列实现方式与效率分析六、队列的应用场景七、Python中队列与优先级队列使用 一、什么是队列 队列是一种特殊的线性表&#xff0c;遵循先入先出…

express习惯养成小程序-计算机毕设 附源码 32209

习惯养成小程序的设计与实现 摘 要 随着我国经济迅速发展&#xff0c;人们对手机的需求越来越大&#xff0c;各种手机软件也都在被广泛应用&#xff0c;但是对于手机进行数据信息管理&#xff0c;对于手机的各种软件也是备受用户的喜爱&#xff0c;习惯养成小程序被用户普遍使…

WebSocket协议在java中的使用

学习的最大理由是想摆脱平庸&#xff0c;早一天就多一份人生的精彩&#xff1b;迟一天就多一天平庸的困扰。各位小伙伴&#xff0c;如果您&#xff1a; 想系统/深入学习某技术知识点… 一个人摸索学习很难坚持&#xff0c;想组团高效学习… 想写博客但无从下手&#xff0c;急需…

Centos7上面部署redis

Centos7上面部署redis 编写这个部署redis&#xff0c;只是为了另一个文章入侵redis做准备&#xff0c;网上还有好多类似的文章&#xff0c;这个单纯的就是部署安装&#xff0c;并简单的测试使用以下 关联其他文章 [1]VMware上面安装部署centos7镜像系统【详细含镜像】 [2]血的教…

美女骑士开箱VELO Angel TT,银色天使,无痛骑行

阳光、女孩、自行车&#xff0c;脸上的笑容或明媚&#xff0c;或神秘&#xff0c;或青涩&#xff0c;在这个时候&#xff0c;世界上没有什么比骑行女孩更美的了&#xff01;      在北京&#xff0c;有一个热爱骑行的女孩&#xff0c;名叫季思铭&#xff0c;目前是中国农业…

CDA一级备考思维导图

CDA一级备考思维导图 第一章 数据分析概述与职业操守1、数据分析概念、方法论、角色2、数据分析师职业道德与行为准则3、大数据立法、安全、隐私 CDA一级复习备考资料共计七个章节&#xff0c;如需资料&#xff0c;请留言&#xff0c;概览如下图&#xff1a; 第一章 数据分析…

【Java】使用IntelliJ IDEA搭建SSM(MyBatis-Plus)框架并连接MySQL数据库

步骤 0 准备工作1 创建Maven项目2 配置Maven依赖3 配置数据源4 项目结构5 创建实体类6 创建数据访问层7 创建服务层8 创建Controller层9 启动项目10 使用Postman测试接口 0 准备工作 下载并安装 IntelliJ IDEA下载并安装 MySQL 数据库下载并安装Postman测试工具使用 Navicat 创…

WebSocket了解

一.什么是WebSocket WebSocket是HTML5下一种新的协议&#xff08;websocket协议本质上是一个基于tcp的协议&#xff09;它实现了浏览器与服务器全双工通信&#xff0c;能更好的节省服务器资源和带宽并达到实时通讯的目的Websocket是一个持久化的协议 二.websocket的原理 web…

Linux4.5、进程状态

个人主页&#xff1a;Lei宝啊 愿所有美好如期而遇 目录 进程状态介绍 Linux下具体进程状态 R状态 和 S状态 D状态 T状态 t状态 Z状态 X状态 进程状态介绍 首先&#xff0c;进程状态有运行&#xff0c;阻塞&#xff0c;挂起&#xff0c;这些只是一个大体的概括&am…

判断 一个整数 是不是 2 的阶次方

问题&#xff1a;判断 一个整数 是不是 2 的阶次方 思路&#xff1a; 1、先用while循环&#xff0c;判断该数字是否大于1 2、大于1&#xff0c;那么进行取模2&#xff0c;判断该数字是否是偶数 3、是偶数&#xff0c;那么除以2&#xff0c;看能不能整除掉&#xff0c;整除到最…

Flutter桌面应用开发之毛玻璃效果

目录 效果实现方案依赖库支持平台实现步骤注意事项话题扩展 毛玻璃效果&#xff1a;毛玻璃效果是一种模糊化的视觉效果&#xff0c;常用于图像处理和界面设计中。它可以通过在图像或界面元素上应用高斯模糊来实现。使用毛玻璃效果可以增加图像或界面元素的柔和感&#xff0c;同…

点赞业务对MySQL和Redis和MongoDB理解

点赞 点赞业务比较频繁,很多人业务可能都会有这个,比如:博客,视频,文章,动态,评论等,但是不应该是核心业务,不应该大量地请求MySQL数据库,给数据库造成大量的资源消耗,MySQL的数据库是非常宝贵的. 以某音为例,当我去搜索的时候,全抖音比较高的点赞数目应该是在1200w - 2000w,…

【视觉SLAM十四讲学习笔记】第三讲——旋转向量和欧拉角

专栏系列文章如下&#xff1a; 【视觉SLAM十四讲学习笔记】第一讲——SLAM介绍 【视觉SLAM十四讲学习笔记】第二讲——初识SLAM 【视觉SLAM十四讲学习笔记】第三讲——旋转矩阵 【视觉SLAM十四讲学习笔记】第三讲——Eigen库 本章将介绍视觉SLAM的基本问题之一&#xff1a;如何…

Unity 自带的一些可以操控时间的属性或方法。

今天来总结下Unity自带的一些可以操控时间的方法。 1、Time.time。比较常用计算运行时间而触发特定事件。 public class Controller : MonoBehaviour {public float eventTime 5f; // 触发事件的时间private float startTime; // 游戏开始的时间private void Start(){startT…