力扣416. 分割等和子集(java 动态规划)

Problem: 416. 分割等和子集

文章目录

题目描述

思路

该题目可以归类为0-1背包问题 ,具体到细节可以再归纳为背包是否装满问题

1.首先判断数组元素和的奇偶性(奇数则不能划分)

2.我们定义一个二维布尔类型数组,用于记录每一阶段的可选状态

3.针对于动态转移方程:我们要判断最终是否可以选取一些数使其和为原来数组元素和的一半 ,即通过一层一层的选择数(状态转移),判断最终状态是否可达(能否有一组数使得其和为原来数组元素和的一半)每一个位置都会有选与不选 两种状态,若选取则dp[i][j] == dp[i - 1][j - nums[i]] ,若不选取则dp[i][j] == dp[i - 1][j] ;则我们可以知道每一个位置的状态只取决于上一层的两个位置的状态即 dp[i][j] == dp[i - 1][j] || dp[i - 1][j - numsi

解题方法

1.获取数组的长度(假设为 n n n),统计数组所有元素的和(假设为 s u m sum sum)并判断若 s u m sum sum为奇数则直接返回false,否则将 s u m sum sum除以2;

2.定义一个Boolean类型的二维数组dp(行数为 n n n,列数为 s u m + 1 sum + 1 sum+1),用于记录每个状态,初始化(将dp[0][0]位置设置为true),并判断若*nums[0]位置值小于sum(注意sum已经除等于2了)*则将dp[0][nums[0]]位置设为true,这样就将第0层的所有状态设置完毕

3.我们从dp数组的第一层开始遍历,并完成动态转移 ,在没有越界(此处笔者感觉也不能就叫做越界,但是思想和上述思路中的动态转移是一致的。执行该if判断只是为了更好契合该dp数组。读者可以尝试画一画dp数组演示一边便可理解 )的情况下(if (j - nums[i] >= 0) )则dp[i][j] = dp[i - 1][j] || dp[i - 1][j - nums[i]];否则dp[i][j] = dp[i - 1][j];

4.返回最终位置的状态即**dp[n - 1][sum]**位置的状态

复杂度

时间复杂度:

O ( M N ) O(MN) O(MN);其中 M M M为数组长度, N N N为数组元素和的一半

空间复杂度:

O ( M N ) O(MN) O(MN);其中 M M M为数组长度, N N N为数组元素和的一半

Code

java 复制代码
class Solution {
    /**
     * Similar to the 0-1 knapsack problem (can the knapsack be filled?)
     *
     * @param nums Given arrays
     * @return boolean
     */
    public boolean canPartition(int[] nums) {
        int n = nums.length;
        int sum = 0;
        for (int i = 0; i < n; ++i) {
            sum += nums[i];
        }
        //Indivisible if odd
        if (sum % 2 == 1) {
            return false;
        }
        sum /= 2;
        //Records whether each node is reachable
        boolean[][] dp = new boolean[n][sum + 1];
        dp[0][0] = true;
        if (nums[0] <= sum) {
            dp[0][nums[0]] = true;
        }
        //State transition equation
        for (int i = 1; i < n; ++i) {
            for (int j = 0; j <= sum; ++j) {
                //Determine whether the line has been crossed
                if (j - nums[i] >= 0) {
                    dp[i][j] = dp[i - 1][j] || dp[i - 1][j - nums[i]];
                } else {
                    dp[i][j] = dp[i - 1][j];
                }
            }
        }
        return dp[n - 1][sum];
    }
}
相关推荐
进击的CJR2 分钟前
redis cluster 部署
java·redis·mybatis
重生之后端学习11 分钟前
19. 删除链表的倒数第 N 个结点
java·数据结构·算法·leetcode·职场和发展
qq_124987075313 分钟前
基于小程序中医食谱推荐系统的设计(源码+论文+部署+安装)
java·spring boot·后端·微信小程序·小程序·毕业设计·计算机毕业设计
Coder_Boy_16 分钟前
基于SpringAI的在线考试系统-阅卷评分模块时序图
java·人工智能·spring boot
linweidong35 分钟前
C++大型系统中如何组织头文件和依赖树?
java·c++·架构
鹿角片ljp42 分钟前
力扣14.最长公共前缀-纵向扫描法
java·算法·leetcode
pengweizhong1 小时前
Dynamic‑SQL2 查询篇:MyBatis 增强利器,让 SQL 像写 Java 一样丝滑
java·sql·教程
Remember_9931 小时前
【数据结构】深入理解优先级队列与堆:从原理到应用
java·数据结构·算法·spring·leetcode·maven·哈希算法
Leo July1 小时前
【Java】Spring Cloud 微服务生态全解析与企业级架构实战
java·spring cloud
Marktowin1 小时前
SpringBoot项目的国际化流程
java·后端·springboot