前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Leetcode-Medium 416. Partition Equal Subset Sum

Leetcode-Medium 416. Partition Equal Subset Sum

作者头像
致Great
发布2019-03-15 10:43:44
4400
发布2019-03-15 10:43:44
举报
文章被收录于专栏:程序生活程序生活

题目描述

给定仅包含正整数的非空数组,查找是否可以将数组划分为两个子集,使得两个子集中的元素总和相等。 例子1:

代码语言:javascript
复制
Input: [1, 5, 11, 5]

Output: true

Explanation: The array can be partitioned as [1, 5, 5] and [11].

例子2:

代码语言:javascript
复制
Input: [1, 2, 3, 5]

Output: false

Explanation: The array cannot be partitioned into equal sum subsets.

思路

如果两个子集中的元素和相等,那么我们至少可以挖掘两个信息:

  • 如果数组为空,那么应该返回False
  • 如果数组元素相加的和为奇数时,应该范围False。 因为两个相等的数相加必为偶数(full_sum=target+target=2*target=2*target) target=sum(nums)//2=sum(nums)>>1(右移一位相当于整除)

思路1 :找出所有可能子集的和,判断 target是否出现在possible_sums 思路2:动态规划:[LeetCode] Partition Equal Subset Sum 相同子集和分割

定义一个一维的dp数组,其中dp[i]表示原数组是否可以取出若干个数字,其和为i。那么我们最后只需要返回dp[target]就行了。初始化dp[0]为true,由于题目中限制了所有数字为正数,那么就不用担心会出现和为0或者负数的情况。关键问题就是要找出状态转移方程了,我们需要遍历原数组中的数字,对于遍历到的每个数字nums[i],需要更新dp数组,我们的最终目标是想知道dp[target]的boolean值,就要想办法用数组中的数字去凑出target,因为都是正数,所以只会越加越大,那么加上nums[i]就有可能会组成区间 [nums[i], target] 中的某个值,那么对于这个区间中的任意一个数字j,如果 dp[j - nums[i]] 为true的话,说明现在已经可以组成 j-nums[i] 这个数字了,再加上nums[i],就可以组成数字j了,那么dp[j]就一定为true。如果之前dp[j]已经为true了,当然还要保持true,所以还要‘或’上自身,于是状态转移方程如下:

dp[j] = dp[j] || dp[j - nums[i]] (nums[i] <= j <= target)

有了状态转移方程,那么我们就可以写出代码了,这里需要特别注意的是,第二个for循环一定要从target遍历到nums[i],而不能反过来,想想为什么呢?因为如果我们从nums[i]遍历到target的话,假如nums[i]=1的话,那么[1, target]中所有的dp值都是true,因为dp[0]是true,dp[1]会或上dp[0],为true,dp[2]会或上dp[1],为true,依此类推,完全使我们的dp数组失效了

代码实现

思路1

代码语言:javascript
复制
class Solution:
    def canPartition(self, nums: List[int]) -> bool:
        if sum(nums) and not sum(nums)%2:
            possible_sums={0}
            for num in nums:
                possible_sums.update({(num+v) for v in possible_sums})
                if sum(nums)>>1 in possible_sums:
                    return True
        return False

思路2

代码语言:javascript
复制
class Solution:
    def canPartition(self, nums: List[int]) -> bool:
        sums=sum(nums)
        target=sums>>1
        if not sums or sums%2:
            return False
        dp=[False]*(target+1)
        dp[0]=True
        for num in nums:
            for i in range(target,num-1,-1):
                dp[i]=dp[i] or dp[i-num]
        return dp[target]
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2019.03.06 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 题目描述
  • 思路
  • 代码实现
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档