动态规划 | part03

416. 分割等和子集 - 力扣(LeetCode)

给定一个只包含正整数的非空数组。是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。

注意: 每个数组中的元素不会超过 100 数组的大小不会超过 200

示例 1:

  • 输入: [1, 5, 11, 5]
  • 输出: true
  • 解释: 数组可以分割成 [1, 5, 5] 和 [11].

示例 2:

  • 输入: [1, 2, 3, 5]
  • 输出: false
  • 解释: 数组不能分割成两个元素和相等的子集.

提示:

  • 1 <= nums.length <= 200

  • 1 <= nums[i] <= 100

    public boolean canPartition(int[] nums) {
    // 1. 空数组检查
    if (nums.length == 0) {
    return false;
    }

    复制代码
      // 2. 计算总和
      int sum = 0;
      for (int i = 0; i < nums.length; i++) {
          sum += nums[i];
      }
      
      // 3. 奇数判断:总和为奇数,无法平分
      if (sum % 2 != 0) {
          return false;
      }
      
      // 4. 目标:背包容量 = 总和的一半
      int target = sum / 2;
      
      // 5. 定义DP数组
      // dp[j] 表示:背包容量为 j 时,能装入的最大物品价值(这里价值=重量=nums[i])
      int[] dp = new int[target + 1];  // 默认初始化为0
      
      // 6. 遍历每个物品(数字)
      for (int i = 0; i < nums.length; i++) {
          
          // 7. 倒序遍历背包容量(关键!必须倒序,防止重复放入同一物品)
          for (int j = target; j >= nums[i]; j--) {
              
              // 8. 状态转移:不放 vs 放,取最大值
              // 不放:dp[j](保持原值)
              // 放:dp[j - nums[i]] + nums[i](腾出空间后的最大价值 + 当前物品价值)
              dp[j] = Math.max(dp[j], dp[j - nums[i]] + nums[i]);
          }
      }
      
      // 9. 判断:背包容量为target时,能否正好装满?
      return dp[target] == target;

    }

解题:

我们目标就是找到sum/ 2是否可以取到,我们的dp数组就代表当前的容量所能装的最大价值。递推公式是dp[j] = Math.max(dp[j], dp[j - nums[i]] + nums[i]),就是考虑放与不放的区别,前者是不放,后者是放入。最后dp[target] == target可以判断此容量的背包的最大价值是否是target,如果是则说明正好被放满了,返回true。

相关推荐
mit6.8242 小时前
合法括号字符串|递归|树
算法
普通网友2 小时前
C++与Rust交互编程
开发语言·c++·算法
逆境不可逃2 小时前
【春节篇】LeetCode 热题 100 之 238.除了自身以外数组的乘积
数据结构·算法·leetcode
铸人2 小时前
再论自然数全加和 - 质数螺旋及其生成程序
数学·算法·数论·复数
散峰而望2 小时前
【算法竞赛】堆和 priority_queue
开发语言·数据结构·c++·算法·贪心算法·动态规划·推荐算法
WarPigs2 小时前
UI显示任务目的地标记的方法
算法·ui
蚊子码农2 小时前
算法题解记录-560和为k的子数组
算法
alexwang2112 小时前
B2007 A + B 问题 题解
c++·算法·题解·洛谷
重生之后端学习3 小时前
46. 全排列
数据结构·算法·职场和发展·深度优先·图论