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

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

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


原创内容 转载请注明出处

分享给朋友:

相关文章

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

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

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

力扣1137题:动态规划解泰波那契数 高效求解第N项的秘密

力扣1137题:动态规划解泰波那契数 高效求解第N项的秘密

一:重新解读题目泰波那契数列是一个充满数学趣味的递推序列:从第3项开始,每个数均为前三个数的和(即Tₙ₊₃ = Tₙ + Tₙ₊₁ + Tₙ₊₂)。当给定整数n时,需要高效计算出第n项的值。面对此类递...

CSP-J 2019纪念品题解(洛谷P5662):动态规划+完全背包问题的实战应用

CSP-J 2019纪念品题解(洛谷P5662):动态规划+完全背包问题的实战应用

一、题目解读2019年CSP-J的“纪念品”问题(对应洛谷P5662)要求玩家在T天内通过买卖纪念品最大化金币收益。每天可交易N种商品,需计算最优策略下的最终金币数。题目强调动态规划思维与资源分配优化...

手搓顺序表类代码注释与详解:从零实现动态数组(新手教程)

一、简介和特点顺序表(Sequential List)是数据结构中基础的一种线性表,其特点是将数据元素存储在连续的内存空间中。通过数组实现,支持随机访问(即通过索引直接访问元素),适用于频繁随机读取的...

牛客12576题解题全解析:动态规划+质因数分解实现跳跃问题最优解

牛客12576题解题全解析:动态规划+质因数分解实现跳跃问题最优解

一、题目解读牛客12576题是一道经典的算法题,要求给定起点N和终点M,求解从N到M的最少跳跃次数。题目考察的核心在于路径优化与动态规划思想,需结合数论中的质因数分解技巧,通过合理设计算法降低时间复杂...

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

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

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

发表评论

访客

看不清,换一张

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