zoukankan      html  css  js  c++  java
  • 58.Partition Equal Subset Sum(判断一个数组是否可以分成和相等的两个数组)

    Level:

      Medium

    题目描述:

    Given a non-empty array containing only positive integers, find if the array can be partitioned into two subsets such that the sum of elements in both subsets is equal.

    Note:

    1. Each of the array element will not exceed 100.
    2. The array size will not exceed 200.

    Example 1:

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

    Example 2:

    Input: [1, 2, 3, 5]
    
    Output: false
    
    Explanation: The array cannot be partitioned into equal sum subsets.
    

    思路分析:

      方法一:利用暴力的dfs进行遍历,找数组中有么有和为sum/2的组合,先看sum是否为偶数,如果为奇数直接false。

      方法二:动态规划的思想。我们定义一个一维的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数组失效了。

    代码:

    思路一:

    public class Solution{
        public boolean canPartition(int []nums){
            int sum=0;
            for(int i=0;i<nums.length;i++){
                sum=sum+nums[i];
            }
            if(sum%2==1)
                return false;
            sum=sum/2;
            Arrays.sort(nums);//排序,方剪枝
            return dfs(0,sum,nums);
        }
        public boolean dfs(int index,int sum,int []nums){
            if(index<nums.length&&nums[index]==sum)
                return true;
            if(index<nums.length&&nums[index]>sum)
                return false;
            return dfs(index+1,sum-nums[index],nums)||dfs(index+1,sum,nums);
        }
    }
    

    思路二:

    public class Solution{
        public boolean canPartition(int []nums){
            int sum=0;
            for(int i=0;i<nums.length;i++){
                sum=sum+nums[i];
            }
            if(sum%2==1)
                return false;
            sum=sum/2;
            boolean []dp=new boolean[sum+1];//dp[i]表示和为i能否由数组中部分元素构成
            Arrays.fill(dp,false);
            dp[0]=true;
            for(int num:nums){
                for(int i=sum;i>=num;i--){
                    dp[i]=dp[i]||dp[i-num];
                }
            }
            return dp[sum];
        }
    }
    
  • 相关阅读:
    图片懒加载技术
    验证码处理
    基于requests模块的cookie,session和线程池爬取
    最快理解
    Golang
    Django REST framework
    Django REST framework
    Django
    Django
    搭建邮件服务器 Postfix + Dovecot (CentOS)
  • 原文地址:https://www.cnblogs.com/yjxyy/p/11089503.html
Copyright © 2011-2022 走看看