那些年那些有趣的数学 2018-02-10
背包问题是一类经典的动态规划问题,本节只介绍两类最简单的背包问题:01背包问题和完全背包问题。
有一类动态规划可解的问题,它可以描述成若干个有序的阶段,且每个阶段的状态只和上一阶段的状态有关,一般把这类问题称为多阶段动态规划问题。如下图所示,该问题被分为 5个阶段,其中状态 F属于阶段 3,它由状态 2的状态 C和状态 D推得。01背包问题就是这样一个例子。
01背包问题是这样的:
下面介绍动态规划的方法。
令 dp[i][v]表示前 i件物品 (1≤i≤n, 0≤v≤V)恰好装入容量为 v的背包中所能获得的最大价值。考虑对第 i件物品的选择策略,有两种策略:
由此可以写出状态转移方程:
$dp[i][v]=max\left \{ dp[i-1][v], dp[i-1][v-w[i]]+c[i] \right \}$
$1\leqslant i\leqslant n,w[i]\leqslant v\leqslant V$
注意到 dp[i][v]只与之前的状态 dp[i-1][]有关,所以可以枚举 i从 1到 n ,v从 0到 V,通过边界 dp[0][v](0≤v≤V)(即前 0件物品放入任何容量 v的背包中都只能获得价值 0)就可以把整个 dp数组递推出来。而由于 dp[i][v]表示的是恰好为 v的情况,所以需要枚举 dp[n][v](0≤v≤V),取最大值才是最后的结果。
代码如下:
1 int i, v; 2 for(i=1; i<=n; ++i) { 3 for(v=w[i]; v<=V; ++v) { 4 dp[i][v] = max(dp[i-1][v], dp[i-1][v-w[i]]+c[i]); 5 } 6 }
此时时间复杂度和空间复杂度都为 O(nV),空间复杂度还可以再优化。
注意到状态转移方程中计算 dp[i][v]时总是只需要 dp[i-1][v]左侧部分的数据,因此可以直接开一个一维数组 dp[v],枚举方向改变为 i从 1到 n, v从 V到 0(逆序!),这样状态转移方程为:
$dp[v]=max(dp[v],dp[v-w[i]]+c[i])$
$1\leqslant i\leqslant n,w[i]\leqslant v\leqslant V$
代码如下:
int i, v; // 滚动数组形式 for(i=; i<=n; ++i) { for(v=V; v >= w[i]; --v) { // 逆序枚举 v dp[v] = max(dp[v], dp[v-w[i]] + c[i]); } }
特别说明:如果用二维数组存放,v的枚举是顺序还是逆序都无所谓;如果使用一维数组存放,则 v的枚举必须是逆序!
完整的求解 01背包问题的代码如下:
1 /* 2 01背包问题 3 */ 4 5 #include <stdio.h> 6 #include <string.h> 7 #include <math.h> 8 #include <stdlib.h> 9 #include <time.h> 10 #include <stdbool.h> 11 12 #define maxn 100 13 #define maxv 1000 14 int w[maxn], c[maxn], dp[maxv]; 15 16 // 求较大值 17 int max(int a, int b) { 18 return (a>b ? a : b); 19 } 20 21 int main() { 22 int i, v, n, V; 23 scanf("%d %d", &n, &V); 24 for(i=1; i<=n; ++i) { 25 scanf("%d", &w[i]); // 输入物品重量 26 } 27 for(i=1; i<=n; ++i) { 28 scanf("%d", &c[i]); // 输入物品价值 29 } 30 // 边界 31 for(v=0; v <= V; ++v) { 32 dp[v] = 0; 33 } 34 // 状态转移方程 35 for(i=1; i<=n; ++i) { 36 for(v=V; v >= w[i]; --v) { // 逆序枚举 v 37 dp[v] = max(dp[v], dp[v-w[i]] + c[i]); 38 } 39 } 40 // 寻找最大值即为答案 41 int maxc = 0; 42 for(v=0; v<=V; ++v) { 43 if(dp[v] >= maxc) { 44 maxc = dp[v]; 45 } 46 } 47 printf("%d\n", maxc); // 输出 48 49 return 0; 50 }
另外,01背包中的每个物品都可以看作一个阶段,这个阶段中的状态有 dp[i][0]~dp[i][v],它们均由上一个阶段的状态得到。事实上,对能够划分阶段的问题来说,都可以尝试把阶段作为状态的一维,这可以使我们更方便地得到无后效性的状态。从中也可以得到这么一个技巧,如果当前设计的状态不满足无后效性,那么不妨把状态进行升维,即增加一维或若干维来表示相应的消息,这样可能就能满足无后效性。
完全背包问题的叙述如下:
同样令 dp[i][v]表示前i件物品恰好放入容量为 v的背包中能获得的最大价值。和 01背包问题一样,考虑对第 i件物品的选择策略,有两种策略:
因此可以写出状态转移方程:
$dp[i][v]=max\left \{ dp[i-1][v], dp[i][v-w[i]]+c[i] \right \}$
$1\leqslant i\leqslant n,w[i]\leqslant v\leqslant V$
边界:dp[0][v]=0(0≤v≤V)
而这个状态转移方程同样可以改写成一维形式,即状态转移方程:
$dp[v]=max(dp[v],dp[v-w[i]]+c[i])$
$1\leqslant i\leqslant n,w[i]\leqslant v\leqslant V$
边界:dp[v]=0(0≤v≤V)
写成一维形式之后和 01背包完全相同,唯一的区别在于这里 v的枚举顺序是正向枚举,而 01背包的一维形式中 v必须是逆向枚举。代码如下:
int i, v; for(i=; i<=n; ++i) { for(v=w[i]; v <= V; ++v) { // 正向枚举 v dp[v] = max(dp[v], dp[v-w[i]]+c[i]); } }
动态规划有时被称为递归的相反的技术。动态规划方案通常使用一个数组来建立一张表,用于存放被分解成众多子问题的解。当算法执行完毕,最终的解法将会在这个表中找到。今天我们先从我们最熟的斐波那契数列数列开始。
#include <iostream> #include <algorithm> #include <cstring> #include <cstdio&am