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

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

4个月前 (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)。代码利用奇偶剪枝和提前终止进一步优化性能,是解决此类问题的经典范式。掌握动态规划的核心逻辑(状态定义、转移方程、边界条件)对解决背包问题、组合问题等具有重要意义。


原创内容 转载请注明出处

分享给朋友:

相关文章

力扣53题:贪心策略与动态规划的完美联姻 三行代码映射算法精髓

力扣53题:贪心策略与动态规划的完美联姻 三行代码映射算法精髓

题目理解在数字的海洋中寻找最具价值的珍珠链:当我们面对一个可能包含正负数的数组时,寻找连续子数组的和最大值就像在波动的股票曲线中捕捉最佳投资时段。问题的核心在于如何处理可能降低总和的负值元素——是忍痛...

NOIP 2008火柴棒等式题解(C++代码实现)  动态规划与枚举算法详解

NOIP 2008火柴棒等式题解(C++代码实现) 动态规划与枚举算法详解

一、题目解读火柴棒等式问题(NOIP 2008,洛谷P1149)要求使用给定数量的火柴棒,构造形如 A + B = C 的等式,其中A、B、C均为整数,且火柴棒总数恰好等于输入值。需统计符合条件的等式...

CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解

CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解

一、题目解读CSP-J 2019年的“公交换乘”题目(洛谷P5661)要求模拟地铁与公交交替出行的费用计算。题目核心在于地铁消费会产生优惠券,而公交可在45分钟内使用优惠券抵扣车费。需要处理n条出行记...

洛谷2789题解:直线交点数的递归求解与优化(附代码详解)

洛谷2789题解:直线交点数的递归求解与优化(附代码详解)

一、题目解读洛谷2789题要求计算n条直线在平面上两两相交时产生的不同交点数量。题目强调“不同”交点,需排除重复情况。解题关键在于如何高效枚举所有可能的交点组合,并避免重复计数。二、解题思路参考代码采...

洛谷P4999题解析:动态规划求解数字拆分与求和问题(附代码)

洛谷P4999题解析:动态规划求解数字拆分与求和问题(附代码)

一、题目解读洛谷P4999题要求处理给定区间 [L, R] 内数字的拆分与求和问题。每个数字需拆分为其各位数字之和,并计算区间内所有数字之和的累加结果。题目需考虑大数情况,并采用取模运算(MOD=1e...

力扣931题最小下降路径和解析 动态规划解法 LeetCode解题技巧

力扣931题最小下降路径和解析 动态规划解法 LeetCode解题技巧

一、题目解读力扣931题「Minimum Falling Path Sum」(最小下降路径和)要求在一个n x n的整数矩阵中,计算从顶部到底部的最小路径和。路径只能从每个位置向下或对角线移动(即向下...

发表评论

访客

看不清,换一张

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