【Leetcode 热题 100】416. 分割等和子集

问题背景

给你一个 只包含正整数非空 数组 n u m s nums nums。请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。

数据约束

  • 1 ≤ n u m s . l e n g t h ≤ 200 1 \le nums.length \le 200 1≤nums.length≤200
  • 1 ≤ n u m s i ≤ 100 1 \le numsi \le 100 1≤numsi≤100

解题过程

要求分成两个子集且和相等,其实就是找到一个总和为 s u m / 2 sum / 2 sum/2 的子集,其中 s u m sum sum 表示整个集合的和。需要注意的是, s u m sum sum 必须是偶数。

此外,和为零是一种可能的状态,所以记忆化数组中的元素要初始化为 − 1 -1 −1。

递归入口 是 d f s ( n − 1 , s u m / 2 ) dfs(n - 1, sum / 2) dfs(n−1,sum/2),表示在 0 , n − 1 0, n - 1 0,n−1 的范围上找到一个和为 s u m / 2 sum / 2 sum/2 的子集。

递归边界 是 d f s ( − 1 , 0 ) = t r u e dfs(-1, 0) = true dfs(−1,0)=true,表示枚举完了整个集合中的所有元素,最终能够使得剩余目标和减少到零。

翻译成递推时要注意,数组下标不能为 − 1 -1 −1,所以要进行转移,将结果映射到 1 , n 1, n 1,n 这个范围上。

具体实现

递归

java 复制代码
class Solution {
    public boolean canPartition(int[] nums) {
        int sum = 0;
        for (int num : nums) {
            sum += num;
        }
        if ((sum & 1) == 1) {
            return false;
        }
        int n = nums.length;
        int[][] memo = new int[n][(sum >> 1) + 1];
        for (int[] row : memo) {
            Arrays.fill(row, -1);
        }
        return dfs(n - 1, sum >> 1, nums, memo);
    }

    private boolean dfs(int i, int j, int[] nums, int[][] memo) {
        if (i < 0) {
            return j == 0;
        }
        if (memo[i][j] != -1) {
            return memo[i][j] == 1;
        }
        boolean res = j >= nums[i] && dfs(i - 1, j - nums[i], nums, memo) || dfs(i - 1, j, nums, memo);
        memo[i][j] = res ? 1 : 0;
        return res;
    }
}

递推

java 复制代码
class Solution {
    public boolean canPartition(int[] nums) {
        int sum = 0;
        for (int num : nums) {
            sum += num;
        }
        if ((sum & 1) == 1) {
            return false;
        }
        sum /= 2;
        int n = nums.length;
        boolean[][] dp = new boolean[n + 1][sum + 1];
        dp[0][0] = true;
        for (int i = 0; i < n; i++) {
            int cur = nums[i];
            for (int j = 0; j <= sum; j++) {
                dp[i + 1][j] = j >= cur && dp[i][j - cur] || dp[i][j];
            }
        }
        return dp[n][sum];
    }
}

空间优化

java 复制代码
class Solution {
    public boolean canPartition(int[] nums) {
        int sum = 0;
        for (int num : nums) {
            sum += num;
        }
        if ((sum & 1) == 1) {
            return false;
        }
        sum /= 2;
        int n = nums.length;
        boolean[] dp = new boolean[sum + 1];
        dp[0] = true;
        // minSum 表示前 i 个数和的上界,不可能超过所有这些数的总和
        int minSum = 0;
        for (int num : nums) {
            // 用这个值来初始化内层循环可以避免许多不必要的判断,想不清楚的话直接用 sum 也可以
            minSum = Math.min(minSum + num, sum);
            for (int j = minSum; j >= num; j--) {
                dp[j] = dp[j] || dp[j - num];
            }
            if (dp[sum]) {
                return true;
            }
        }
        return false;
    }
}
相关推荐
烬羽1 小时前
字符串算法入门:从反转字符串到回文判断,面试不再慌
算法·面试
先吃饱再说17 小时前
判断回文字符串,从一行代码到双指针优化
算法
黄敬峰20 小时前
深入理解算法核心:从递归思想、数组扁平化到快速排序
算法
得物技术21 小时前
从狂野代码到按目标生产:得物推荐 AI Harness 的工程化实践|AICon 演讲整理
人工智能·算法·架构
AI小老六1 天前
SkillOpt 架构拆解:把 Skill 文本当参数,用执行轨迹训练 Agent
后端·算法·ai编程
胡萝卜术1 天前
从“分数打架”到“排名投票”:为什么你的ChatBI必须用RRF?
算法·设计模式·面试
Asize1 天前
初识DFS 与 BFS:递归、队列与图遍历
算法
罗西的思考2 天前
机器人 / 强化学习】HIL-SERL:人类在环驱动的具身智能进化框架
人工智能·算法·机器学习
CSharp精选营2 天前
关系型 vs 非关系型:从原理到选型,一文搞定数据库核心分类
数据结构·nosql·关系型数据库·非关系型数据库·技术选型