分割等和子集

416. 分割等和子集

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

示例 1:

复制代码
输入:nums = [1,5,11,5]
输出:true
解释:数组可以分割成 [1, 5, 5] 和 [11] 。

示例 2:

复制代码
输入:nums = [1,2,3,5]
输出:false
解释:数组不能分割成两个元素和相等的子集。

提示:

  • 1 <= nums.length <= 200
  • 1 <= nums[i] <= 100

1、自己写的shit山代码;

java 复制代码
class Solution {
    public boolean canPartition(int[] nums) {
        int len = nums.length,sum = 0;
        if(len==1)return false;
        for(int i : nums){
            sum+=i;
        }
        if(sum%2!=0)return false;
        int target = sum/2;//计算背包容量

        int[][] dp = new int[len][target+1];//把容量为0的情况也算上
        for(int j = 1;j < target+1;j++){//初始化背包
            if(nums[0]>j){//判断当前容量是否装得下第一个物品
                dp[0][j] = 0;
            }else{
                dp[0][j] = nums[0]; 
            } 
        }

        for(int i = 1;i < len;i++){
            for(int j = 1;j < target+1;j++){//计算最大值
                int a = Math.max(dp[i-1][j],j-nums[i]>=0?dp[i-1][j-nums[i]]+nums[i]:0);
                int b = Math.min(dp[i-1][j],j-nums[i]>=0?dp[i-1][j-nums[i]]+nums[i]:0);
                if(a>target){//判断是否超出界限
                    dp[i][j] = b;
                }else{
                    dp[i][j] = a;
                }
            }
        }

        for(int i = 0;i < len;i++){//寻找是否有答案
            if(dp[i][target]==target){
                return true;
            }
        }
        return false;

    }
}

2、三叶的题解

java 复制代码
class Solution {
    public boolean canPartition(int[] nums) {
        int n = nums.length;

        //「等和子集」的和必然是总和的一半
        int sum = 0;
        for (int i : nums) sum += i;
        int target = sum / 2;

        // 对应了总和为奇数的情况,注定不能被分为两个「等和子集」
        if (target * 2 != sum) return false;

        // 将「物品维度」取消
        int[] f = new int[target + 1];
        for (int i = 0; i < n; i++) {
            int t = nums[i];
            // 将「容量维度」改成从大到小遍历
            for (int j = target; j >= 0; j--) {
                // 不选第 i 件物品
                int no = f[j];
                // 选第 i 件物品
                int yes = j >= t ? f[j-t] + t : 0;
                f[j] = Math.max(no, yes);
            }
        }
        // 如果最大价值等于 target,说明可以拆分成两个「等和子集」
        return f[target] == target;
    }
}
相关推荐
MicroTech20256 分钟前
MLGO微算法科技通过 Lindbladians 设计线性微分方程的近似最优量子算法——开放量子系统框架下的量子ODE求解新范式
科技·算法·量子计算
知乎的哥廷根数学学派12 分钟前
基于多尺度特征提取和注意力自适应动态路由胶囊网络的工业轴承故障诊断算法(Pytorch)
开发语言·网络·人工智能·pytorch·python·算法·机器学习
源代码•宸14 分钟前
Leetcode—85. 最大矩形【困难】
经验分享·算法·leetcode·职场和发展·golang·单调栈
郝学胜-神的一滴23 分钟前
《机器学习》经典教材全景解读:周志华教授匠心之作的技术深探
数据结构·人工智能·python·程序人生·机器学习·sklearn
平哥努力学习ing26 分钟前
《数据结构》-第八章 排序
数据结构·算法·排序算法
CoovallyAIHub27 分钟前
为AI装上“纠偏”思维链,开源框架Robust-R1显著提升多模态大模型抗退化能力
深度学习·算法·计算机视觉
小棠师姐36 分钟前
随机森林原理与实战:如何解决过拟合问题?
算法·机器学习·随机森林算法·python实战·过拟合解决
范纹杉想快点毕业1 小时前
欧几里得算法与扩展欧几里得算法,C语言编程实现(零基础全解析)
运维·c语言·单片机·嵌入式硬件·算法
f***24111 小时前
Bug悬案:技术侦探的破案指南
算法·bug
Swift社区1 小时前
LeetCode 472 连接词
算法·leetcode·职场和发展