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

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

1天前

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


原创内容 转载请注明出处

分享给朋友:

相关文章

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

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

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

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

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

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

LeetCode 120题三角形最小路径和最优解法:动态规划详解与代码实现

LeetCode 120题三角形最小路径和最优解法:动态规划详解与代码实现

一、题目解读LeetCode 120题“三角形最小路径和”要求给定一个由数字组成的三角形,从顶部开始向下移动,每次可向左或向右移动一格,计算从顶至底的最小路径和。三角形以二维向量形式给出,每层元素数量...

洛谷P4551题解题报告:图论与Trie树优化异或路径问题的实战解析

洛谷P4551题解题报告:图论与Trie树优化异或路径问题的实战解析

一、题目解读洛谷P4551题要求在一个无向图中,寻找任意两点路径权值异或后的最大值。题目输入为图的边信息(点数n和n-1条边),每条边包含起点、终点及权值。需输出所有路径中权值异或的最大值。问题核心在...

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

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

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

2022 CSP-J 上升点序(洛谷P8816)解题报告:动态规划求解最长上升序列

2022 CSP-J 上升点序(洛谷P8816)解题报告:动态规划求解最长上升序列

一、题目解读2022年CSP-J题目“上升点序”(洛谷P8816)要求给定一个平面点集,每个点的坐标(x,y)均为整数。题目需要构造一个最长的上升序列,序列中相邻点的坐标满足x和y均严格递增。允许使用...

发表评论

访客

看不清,换一张

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