力扣416-分割等和子集(Java详细题解)

2024-09-06 15:04

本文主要是介绍力扣416-分割等和子集(Java详细题解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目链接:416. 分割等和子集 - 力扣(LeetCode)

前情提要:

因为本人最近都来刷dp类的题目所以该题就默认用dp方法来做。

最近刚学完01背包,所以现在的题解都是以01背包问题为基础再来写的。

如果大家不懂01背包的话,建议可以去学一学,01背包问题可以说是背包问题的基础。

如果大家感兴趣,我后期可以出一篇专门讲解01背包问题。

dp五部曲。

1.确定dp数组和i下标的含义。

2.确定递推公式。

3.dp初始化。

4.确定dp的遍历顺序。

5.如果没有ac打印dp数组 利于debug。

每一个dp题目如果都用这五步分析清楚,那么这道题就能解出来了。

这里下文统一使用一维dp数组。

题目思路:

题目还是比较好入手,判断数组是否可以分为俩个自己,并使俩个子集的元素和相等。

由此可见,如果一个数组他的元素和为偶数的话,就可以使俩个子集的元素和相等。

如果为奇数,就不可能使俩个子集元素和相等,直接return false。

这只是一个小的优化。我们来看看核心思路。

分成俩个子集后,如果我们确定了一个子集的元素和为整个数组和的一半,另一半的就可以确定下来了。

所以我们分析其中的一个子集就可以。

如果一个子集的元素和等于整个数组的元素和的一半,那么这个子集就可以确定下来,整个数组就可以分为俩个子集。

怎么确定个子集的元素和等于整个数组的元素和的一半呢?

这里就要用到01背包的应用了。

01背包是有n个物品,每个物品有其对应的价值和重量,给你一个容量为m的背包,让你求该背包所能放下物品的最大价值。

那么这道题的子集怎么跟01背包关联呢?

其实这个子集的元素和就可以当做01背包的背包容量,元素本身可以当做他的价值和重量,即他的价值和重量都一致。

怎么理解呢?

我们要判断一个集合里是否有数能累加起来等于整个数组元素和的一半。

元素和当做背包容量。

元素本身就是物品。

元素的值就是物品的价值,同时也代表它所占的背包容量。

其实01背包我们是要尽可能的将背包装满然后得出他的最大价值。

那这个子集的元素和就是我们尽可能的将背包装满判断他的最大价值等于容量即可。

接着我们用dp五部曲来系统分析。

1.确定dp数组和i下标的含义。

dp[i] 表示i容量的背包所能放下的最大价值。

如果后面有不理解的,多理解一下dp数组的含义。

2.确定递推公式。

我们每个元素只有选和不选俩种状态。

即选择nums[i]这个元素时,对于背包只有选和不选俩个状态。

我们类比一下01背包问题的递推公式dp[j] = Math.max(dp[j],dp[j - weight[i] + value[i]);

没选的状态就是dp[j]。没有选他那他的重量肯定就不变。

选的状态就是dp[j - weigtht[i]] + value[i]。

既然选择要加nums[i],那我们肯定要求出在放入他之前的最大价值再加上他自身的价值就是背包整个的价值。

放入之前的容量就是j - weigth[i]

因为本题是重量与价值一致。

我们尽可能的将背包填满使它的价值等于它背包的容量。如果等于则return true反之return false

所以当背包容量为j时,dp[j] = Math.max(dp[j],dp[j - nums[i]] + nums[i]);

3.dp初始化。

初始化也很重要,当我们的元素和为0时,那我们所占的元素价值为多少?

肯定为0啊,所以dp[0] = 0。

那其他非零元素和我们怎么初始化呢?

其实非零元素和我们可以不管他们,他们都可以根据当前元素选或不选来推出元素和的价值。

4.确定dp的遍历顺序。

背包问题我们的遍历顺序是先遍历物品再遍历背包,同理,元素和也是先遍历物品(元素)再遍历背包(元素和)。

且第一层循环遍历物品是从前往后遍历,而第二层循环是从后往遍历。

从后往前遍历背包容量是确保每个物品只放一次,即不需要前面的状态。

如果从前往后遍历背包容量可能导致一个物品放入多次,他使用了前面的状态。

而本题元素只能使用一次,所以背包容量应该从后往前遍历。

5.如果没有ac打印dp数组 利于debug。

最后dp[i]模拟后情况就是这样,大家可以打印dp数组对着看看哪与你的不符。

在这里插入图片描述

最终代码:

class Solution {public boolean canPartition(int[] nums) {//定义元素和int sum = 0;for(int i = 0;i < nums.length; i++){sum += nums[i];}//判断如果不为偶数 直接返回falseif(sum % 2 != 0)return false;//目标元素和int tagert = sum / 2;//定义dp数组int[] dp = new int [tagert + 1];//初始化dp数组dp[0] = 0;//确定dp数组的遍历顺序for(int i = 0;i < nums.length;i ++){for(int j = tagert;j >= nums[i];j --){dp[j] = Math.max(dp[j],dp[j - nums[i]] + nums[i]);//如果背包价值等于背包容量 直接return trueif(dp[j] == tagert){return true;}}}return false;}
}

背包问题就是这样,思路分析一大堆,实际代码一小堆。

我们多做多理解就好。

这一篇博客就到这了,如果你有什么疑问和想法可以打在评论区,或者私信我。

我很乐意为你解答。那么我们下篇再见!

这篇关于力扣416-分割等和子集(Java详细题解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1142335

相关文章

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完

Java实现远程执行Shell指令

《Java实现远程执行Shell指令》文章介绍使用JSch在SpringBoot项目中实现远程Shell操作,涵盖环境配置、依赖引入及工具类编写,详解分号和双与号执行多指令的区别... 目录软硬件环境说明编写执行Shell指令的工具类总结jsch(Java Secure Channel)是SSH2的一个纯J

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

SpringBoot 获取请求参数的常用注解及用法

《SpringBoot获取请求参数的常用注解及用法》SpringBoot通过@RequestParam、@PathVariable等注解支持从HTTP请求中获取参数,涵盖查询、路径、请求体、头、C... 目录SpringBoot 提供了多种注解来方便地从 HTTP 请求中获取参数以下是主要的注解及其用法:1

HTTP 与 SpringBoot 参数提交与接收协议方式

《HTTP与SpringBoot参数提交与接收协议方式》HTTP参数提交方式包括URL查询、表单、JSON/XML、路径变量、头部、Cookie、GraphQL、WebSocket和SSE,依据... 目录HTTP 协议支持多种参数提交方式,主要取决于请求方法(Method)和内容类型(Content-Ty

深度解析Java @Serial 注解及常见错误案例

《深度解析Java@Serial注解及常见错误案例》Java14引入@Serial注解,用于编译时校验序列化成员,替代传统方式解决运行时错误,适用于Serializable类的方法/字段,需注意签... 目录Java @Serial 注解深度解析1. 注解本质2. 核心作用(1) 主要用途(2) 适用位置3