分割等和子集

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;
    }
}
相关推荐
EterNity_TiMe_8 分钟前
【论文复现】(CLIP)文本也能和图像配对
python·学习·算法·性能优化·数据分析·clip
机器学习之心19 分钟前
一区北方苍鹰算法优化+创新改进Transformer!NGO-Transformer-LSTM多变量回归预测
算法·lstm·transformer·北方苍鹰算法优化·多变量回归预测·ngo-transformer
yyt_cdeyyds29 分钟前
FIFO和LRU算法实现操作系统中主存管理
算法
daiyang123...1 小时前
测试岗位应该学什么
数据结构
alphaTao1 小时前
LeetCode 每日一题 2024/11/18-2024/11/24
算法·leetcode
kitesxian1 小时前
Leetcode448. 找到所有数组中消失的数字(HOT100)+Leetcode139. 单词拆分(HOT100)
数据结构·算法·leetcode
VertexGeek2 小时前
Rust学习(八):异常处理和宏编程:
学习·算法·rust
石小石Orz2 小时前
Three.js + AI:AI 算法生成 3D 萤火虫飞舞效果~
javascript·人工智能·算法
jiao_mrswang3 小时前
leetcode-18-四数之和
算法·leetcode·职场和发展
qystca3 小时前
洛谷 B3637 最长上升子序列 C语言 记忆化搜索->‘正序‘dp
c语言·开发语言·算法