labuladong 经典动态规划:完全背包问题
NOTE:
一、原题: LeetCode 518. 零钱兑换 II 中等
二、需要和 LeetCode 322. 零钱兑换 进行对比才能够更好地掌握两个问题背后的本质:
1、在 labuladong 动态规划详解 中 对 LeetCode 322. 零钱兑换 进行了分析。
2、可以看到,两个问题都让硬币个数是无限多个,这就让每个子问题都相互独立了,这是在 labuladong 动态规划详解 中分析过的,这是非常重要的一个性质,也是理解后续实现的一个前提。
三、通过这个题,是可以看出01背包和完全背包的差异的,参见下面的"第三步,根据「选择」,思考状态转移的逻辑"段,其中对此有着很好的分析
零钱兑换 2 是另一种典型背包问题的变体,我们前文已经讲了 经典动态规划:0-1 背包问题 和 背包问题变体:相等子集分割。
本文聊的是 LeetCode 第 518 题 Coin Change 2,题目如下:
int change(int amount, int[] coins);
PS:至于 Coin Change 1,在我们前文 动态规划套路详解 写过。
我们可以把这个问题转化为背包问题的描述形式:
有一个背包,最大容量为amount
,有一系列物品coins
,每个物品的重量为coins[i]
,每个物品的数量无限。请问有多少种方法,能够把背包恰好装满?
这个问题和我们前面讲过的两个背包问题,有一个最大的区别就是,每个物品的数量是无限的,这也就是传说中的「完全背包问题」,没啥高大上的,无非就是状态转移方程有一点变化而已。
下面就以背包问题的描述形式,继续按照流程来分析。
解题思路
第二步要明确dp数组的定义
NOTE:
需要对base case进行明确,其实只需要把我"装满"的含义即可,在 labuladong 经典动态规划:0-1背包问题的变体 中已经讨论过了:
显然,背包容量为0,对应的是装满,否则就是没有装满,因此:
dp[0][..] = 0, dp[..][0] = 1
。
经过以上的定义,可以得到:
base case 为dp[0][..] = 0, dp[..][0] = 1
。因为如果不使用任何硬币面值,就无法凑出任何金额;如果凑出的目标金额为 0,那么“无为而治”就是唯一的一种凑法。
第三步,根据「选择」,思考状态转移的逻辑
注意,我们这个问题的特殊点在于物品的数量是无限的,所以这里和之前写的背包问题文章有所不同。
NOTE:
一、上述所说的正是01背包和完全背包的差异所在,具体体现在:
如果你把这第
i
个物品装入了背包1、01背包: 每个物品只能够被选一次,因此
dp[i][j]
应该等于dp[i-1][j-coins[i-1]]
。2、完全背包: 每个物品只能够被选无数次,因此
dp[i][j]
应该等于dp[i][j-coins[i-1]]
。这是两者的显著差异。
如果你不把这第i
个物品装入背包,也就是说你不使用coins[i]
这个面值的硬币,那么凑出面额j
的方法数dp[i][j]
应该等于dp[i-1][j]
,继承之前的结果。
如果你把这第i
个物品装入了背包,也就是说你使用coins[i]
这个面值的硬币,那么dp[i][j]
应该等于dp[i][j-coins[i-1]]
。
实现
综上就是两种选择,而我们想求的dp[i][j]
是「共有多少种凑法」,所以dp[i][j]
的值应该是以上两种选择的结果之和:
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= amount; j++) {
if (j - coins[i-1] >= 0)
dp[i][j] = dp[i - 1][j]
+ dp[i][j-coins[i-1]];
return dp[N][W]
最后一步,把伪码翻译成代码,处理一些边界情况。
我用 Java 写的代码,把上面的思路完全翻译了一遍,并且处理了一些边界问题:
int change(int amount, int[] coins) {
int n = coins.length;
int[][] dp = amount int[n + 1][amount + 1];
// base case
for (int i = 0; i <= n; i++)
dp[i][0] = 1;
for (int i = 1; i <= n; i++) {
for (int j = 1; j <= amount; j++)
if (j - coins[i-1] >= 0)
dp[i][j] = dp[i - 1][j]
+ dp[i][j - coins[i-1]];
else
dp[i][j] = dp[i - 1][j];
}
return dp[n][amount];
}
而且,我们通过观察可以发现,dp
数组的转移只和dp[i][..]
和dp[i-1][..]
有关,所以可以压缩状态,进一步降低算法的空间复杂度:
int change(int amount, int[] coins) {
int n = coins.length;
int[] dp = new int[amount + 1];
dp[0] = 1; // base case
for (int i = 0; i < n; i++)
for (int j = 1; j <= amount; j++)
if (j - coins[i] >= 0)
dp[j] = dp[j] + dp[j-coins[i]];
return dp[amount];
}
这个解法和之前的思路完全相同,将二维dp
数组压缩为一维,时间复杂度 O(N*amount),空间复杂度 O(amount)。
至此,这道零钱兑换问题也通过背包问题的框架解决了。