377. 组合总和 Ⅳ

  1. 确定dp数组以及下标的含义

dp[i]: 凑成目标正整数为i的排列个数为dp[i]

  1. 确定递推公式

dp[i] += dp[i - nums[j]];

  1. dp数组如何初始化

因为递推公式dp[i] += dp[i - nums[j]]的缘故,dp[0]要初始化为1,这样递归其他dp[i]的时候才会有数值基础。
至于dp[0] = 1 有没有意义呢?
其实没有意义,所以我也不去强行解释它的意义了,因为题目中也说了:给定目标值是正整数! 所以dp[0] = 1是没有意义的,仅仅是为了推导递推公式。

  1. 确定遍历顺序
  2. 个数可以不限使用,说明这是一个完全背包。

得到的集合是排列,说明需要考虑元素之间的顺序。
如果求组合数就是外层for循环遍历物品,内层for遍历背包
如果求排列数就是外层for遍历背包,内层for循环遍历物品
target(背包)放在外循环,将nums(物品)放在内循环,内循环从前到后遍历

  1. 举例来推导dp数组

我们再来用示例中的例子推导一下:
image.png

C++测试用例有两个数相加超过int的数据,所以需要在if里加上dp[i] < INT_MAX - dp[i - num]。

  1. class Solution {
  2. public:
  3. int combinationSum4(vector<int>& nums, int target) {
  4. vector<int> dp(target + 1, 0);
  5. dp[0] = 1;
  6. for (int i = 0; i <= target; i++) { // 遍历背包
  7. for (int j = 0; j < nums.size(); j++) { // 遍历物品
  8. if (i - nums[j] >= 0 && dp[i] < INT_MAX - dp[i - nums[j]]) {
  9. dp[i] += dp[i - nums[j]];
  10. }
  11. }
  12. }
  13. return dp[target];
  14. }
  15. };