当前位置:首页 > 力扣 > LeetCode 416题解法:分割等和子集的动态规划解析(附C++代码)

LeetCode 416题解法:分割等和子集的动态规划解析(附C++代码)

8个月前 (07-19)

LeetCode 416题解法:分割等和子集的动态规划解析(附C++代码) 动态规划 背包问题 力扣题解 状态转移 C++ 第1张

一、题目解读

LeetCode 416题“分割等和子集”要求判断一个整数数组能否被分成两个元素和相等的子集。例如,数组[1,5,11,5]可以分成[1,5,5]和[11],返回true;而[1,2,3]无法等分,返回false。问题本质是子集求和的变种,需判断是否存在子集使其和等于数组总和的一半。

二、解题思路

采用动态规划解决。核心思想是构建一个布尔型dp数组,dp[i]表示能否用原数组元素组成和为i的子集。关键在于状态转移方程:若当前元素num加入子集后,剩余元素能组成目标值target-num,则dp[target]为真。为避免重复使用元素,内层循环需反向遍历。

三、解题步骤

1. 计算数组元素总和sum,若为奇数直接返回false(无法等分)。

2. 定义目标值target = sum/2,初始化dp[0]=true(空子集和为0)。

3. 外层循环遍历每个元素num,内层从target反向遍历至num:

    若dp[j](j为目标值)或dp[j-num]为真,更新dp[j] = true。

    若提前发现dp[target] = true,立即终止循环返回true。

4. 最终返回dp[target]的结果。

四、代码与注释

class Solution {
public:
    bool canPartition(vector<int>& nums) {
        int sum = accumulate(nums.begin(), nums.end(), 0); // 计算总和
        if (sum % 2!= 0) return false; // 奇数和无法等分

        int target = sum / 2; // 目标值
        int n = nums.size();
        vector<bool> dp(target + 1, false); // dp[i]:能否组成和为i的子集
        dp[0] = true; // 和为0的子集始终存在

        for (int num : nums) { // 遍历每个元素
            for (int j = target; j >= num; j--) { // 反向遍历避免重复使用元素
                dp[j] = dp[j] || dp[j - num]; // 状态转移:若j或j-num可达,则j可达
            }
            if (dp[target]) return true; // 提前终止条件
        }
        return dp[target];
    }
};

注释解析:

● accumulate函数简化了总和计算,奇偶判断为关键剪枝条件。

● 反向循环保证每个num仅被使用一次,避免组合重复。

● 提前终止优化减少无效遍历,提升效率。

五、总结

动态规划通过状态压缩将子集问题转化为线性求解,时间复杂度O(n*target),空间复杂度O(target)。代码利用奇偶剪枝和提前终止进一步优化性能,是解决此类问题的经典范式。掌握动态规划的核心逻辑(状态定义、转移方程、边界条件)对解决背包问题、组合问题等具有重要意义。


原创内容 转载请注明出处

分享给朋友:

相关文章

70.爬楼梯|三步破解动态规划核心奥秘

70.爬楼梯|三步破解动态规划核心奥秘

题意新解:站在楼梯底仰望n级台阶,每步可选1或2阶,最终的路径组合犹如斐波那契数列般展开。比如到达第3阶的路径可由第1阶跨两步,或第2阶跨一步构成,这种递推规律揭示了两两相邻状态间的紧密关联。思路解析...

力扣740.删除并获得点数 预处理与动态规划的巧妙融合

力扣740.删除并获得点数 预处理与动态规划的巧妙融合

题意解析:给定一组数字,每当你选择一个数字x时,所有等于x-1和x+1的数字都会被自动移除。你需要通过巧妙的选择顺序,最大化获得的点数总和。这个问题可以转化为对离散化数字分布的动态规划问题——将相邻数...

【动态规划入门】力扣509题:斐波那契数列的经典解法与优化思路

【动态规划入门】力扣509题:斐波那契数列的经典解法与优化思路

题目解读‌斐波那契数列是一个经典的数学问题,在计算机科学中常被用作算法教学的入门案例。这个神奇的数列从0和1开始,后续每个数字都是前两个数字之和。题目要求我们计算第n个斐波那契数,看似简单的问题背后却...

GESP2023年六级真题解析:动态规划解决小杨买饮料问题(洛谷3873)

GESP2023年六级真题解析:动态规划解决小杨买饮料问题(洛谷3873)

一、题目解读小杨买饮料是GESP 2023年六级认证考试中的一道经典动态规划题目,考察学生对背包问题的理解和应用能力。题目描述小杨需要购买n种饮料,每种饮料有特定的体积w和价格v,他要在不超过容量l的...

洛谷P10472题解:利用栈求解最长有效括号

洛谷P10472题解:利用栈求解最长有效括号

一、题目解读洛谷P10472题要求计算给定字符串中最长有效括号的长度。有效括号指括号成对匹配(如"()[]{}"),子串需连续且内部嵌套正确。题目核心在于判断括号匹配的连续性,并找...

牛客4580题解:动态规划求解网格路径概率问题(C++代码实现)

牛客4580题解:动态规划求解网格路径概率问题(C++代码实现)

一、题目解读牛客4580题要求在一个n×m的网格中计算从起点(1,1)到终点(n,m)的概率。网格中存在障碍物(标记为坏点),路径只能向右或向下移动。到达终点时,若处于边界位置,概率转移规则不同:下边...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。