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

news2024/9/22 6:54:26

题目链接: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];
        }
        //判断如果不为偶数 直接返回false
        if(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 true
                if(dp[j] == tagert){
                    return true;
                }
            }
        }
        return false;
    }
}

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

我们多做多理解就好。

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

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2115485.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

zabbix6.4连接邮箱发出警告

添加告警媒介 默认接收人: 故障级别:{TRIGGER.STATUS}。 服务器:【{HOSTNAME1} 】 发生:{TRIGGER.NAME} 故障! 注:默认接收人:相当于邮件的主题 默认信息:邮件的主题 告警主机:{HOSTNAME1} 告警时间:{EVENT.DATE} {EVENT.TIME} 告警等级:{TRIGGER.SEVERITY} 告警信息:{TRIGGER.…

HTML5中canvas绘图基础详解

第7章 HTML5绘图基础 H5中新增了重要元素canvas,通过绘制任意图形&#xff0c;借助自带API&#xff0c;通过编写js可以控制各种图形&#xff0c;制作动画效果&#xff0c;对web具有划时代意义。 7.1 画布的基础知识 绘图三步骤&#xff1a; 步骤一&#xff1a;使用canvas创…

FreeRTOS学习笔记(五)任务进阶篇

文章目录 前言一、列表和列表项1.1 xList 和 xLIST_ITEM1.2 相关API函数1.3 任务就绪列表 二、任务调度器的启动过程2.1 PendSV 和 SysTick 寄存器2.2 prvStartFirstTask( )2.3 xPortStartScheduler( )2.4 vTaskStartScheduler( ) 的整体流程 三、任务切换3.1基于 SysTick 中断…

一名优秀的工程师应该学会在工作中提升自己,面试篇

xxx 进行 xxx 操作&#xff0c;为什么不行&#xff1f;有人知道吗&#xff1f; 此时&#xff0c;[黑人脸问好号.jpg]。 这里大家可以阅读下《提问的艺术》这本书&#xff0c;这是一本教你如何通过富有技巧性的提问来提高沟通效率并提升自身影响力的书。 Github 上一些开源项目…

zabbix6.4连接钉钉发出警告

zabbix6.4配置钉钉告警 注册钉钉 建一个内部群 添加自定义机器人 配置zabbix服务端 打开脚本告警的配置 # vim /etc/zabbix/zabbix_server.conf AlertScriptsPath/usr/lib/zabbix/alertscripts 准备脚本 安装一个依赖包 # dnf -y install python3-requests # vim /usr/li…

希尔排序/选择排序

前言&#xff1a; 本篇主要对常见的排序算法进行简要分析&#xff0c;代码中均以数组 arr[] { 5, 3, 9, 6, 2, 4, 7, 1, 8 } 为例&#xff0c;进行升序排列。 常见的排序算法有如下&#xff1a; 选择排序中&#xff0c;直接选择排序没有任何实际与教育意义&#xff0c;而堆排…

PopupInner源码分析 -- ant-design-vue系列

PopupInner源码分析 – ant-design-vue系列 1 综述 上一篇讲解了vc-align的工作原理&#xff0c;也就是对齐是如何完成的。这一篇主要讲述包裹 Align的组件&#xff1a;PopupInner组件是如何工作的。 PopupInner主要是对动画状态的管理&#xff0c;比如打开弹窗的时候&#…

【Hot100】LeetCode—763. 划分字母区间

目录 1- 思路哈希表 双指针 2- 实现⭐763. 划分字母区间——题解思路 3- ACM 实现 原题链接&#xff1a;763. 划分字母区间 1- 思路 哈希表 双指针 ① 找到元素最远的出现位置&#xff1a;哈希表② 根据最远出现位置&#xff0c;判断区间的分界线&#xff1a;双指针 实现 …

Java类和对象(详解)

前言&#xff1a; Java中类和对象是比较重要的一章&#xff0c;这一章可以让我们深刻认识到Java语言的"精妙之处"&#xff0c;它不像C语言那么"细"&#xff0c;也不想其他语言封装的那么"保守"。 游刃有余的解决一系列面向对象问题。 面向对象的…

数据集 视线估计-unityeyes-合成数据 >> DataBall

视线估计-合成数据-三维建模-人工智能unityeyes 人眼视线估计仿真合成数据集 inproceedings{wood2016_etra, title {Learning an Appearance-Based Gaze Estimator from One Million Synthesised Images}, author {Wood, Erroll and Baltru{\v{s}}aitis, Tadas and Morency,…

如何使div居中?CSS居中终极指南

前言 长期以来&#xff0c;如何在父元素中居中对齐一个元素&#xff0c;一直是一个让人头疼的问题&#xff0c;随着 CSS 的发展&#xff0c;越来越多的工具可以用来解决这个难题&#xff0c;五花八门的招式一大堆&#xff0c;这篇博客&#xff0c;旨在帮助你理解不同的居中方法…

【电子通识】半导体工艺——保护晶圆表面的氧化工艺

在文章【电子通识】半导体工艺——晶圆制造中我们讲到晶圆的一些基础术语和晶圆制造主要步骤&#xff1a;制造锭(Ingot)、锭切割(Wafer Slicing)、晶圆表面抛光(Lapping&Polishing)。 那么其实当晶圆暴露在大气中或化学物质中的氧气时就会形成氧化膜。这与铁(Fe)暴露在大气…

MySQL record 02 part

查看已建数据库的基本信息&#xff1a; show CREATE DATABASE mydb; 注意&#xff0c;是DATABASE 不是 DATABASEs&#xff0c; 命令成功执行后&#xff0c;回显的信息有&#xff1a; CREATE DATABASE mydb /*!40100 DEFAULT CHARACTER SET utf8mb3 / /!80016 DEFAULT ENCRYPTIO…

基于Python+大数据爬虫+数据可视化大屏的耳机信息的爬取与分析平台设计和实现(2025最新优质项目-系统+源码+部署文档)

博主介绍&#xff1a;✌全网粉丝50W,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和学生毕业项目实战,高校老师/讲师/同行前辈交流✌ 技术范围&#xff1a;SpringBoot、Vue、SSM、HLM…

新手入门Python:Python类中自带的装饰器详解与应用

文章目录 📖 介绍 📖🏡 演示环境 🏡📒 文章内容 📒📝 什么是装饰器?📝 常用装饰器详解📝 高级装饰器📝 综合应用示例⚓️ 相关链接 ⚓️📖 介绍 📖 在Python编程中,有一类特别的工具,它们可以改变或增强函数和方法的行为。这些工具被称为装饰器。对…

使用宝塔面板安装mrdoc

使用宝塔面板安装mrdoc 1、所需环境2、ubuntu系统安装3、宝塔面板安装4、NginxPHPMySQL安装5、python项目管理器安装6、 python版本安装7、mrdoc的部署7.1、下载项目源码7.2、新建python管理器项目 8、使用MySQL作为默认数据库8.1、安装mysqlclient插件8.2、配置数据库连接信息…

qt多线程的两种方法run和movetothread

qt多线程的有什么用&#xff1f; 将耗时长的操作丢入专属线程执行&#xff0c;这样就不会影响主线程的界面操作&#xff0c;操作完再用信号槽等的方式返回结果 1.界面和部件相关都必须在主界面运行&#xff0c;不要用子线程调用或者操作&#xff0c;会引起奇怪的bug&#xff…

推荐一款免费使用的电脑笔记软件,工作必备

今天为大家介绍一款开源的笔记软件——Beaver Notes&#xff08;海狸笔记&#xff09;。 海狸笔记&#xff08;Beaver Notes&#xff09;是一款注重隐私保护的免费、开源且无广告的笔记工具。它拥有一个干净且吸引人的用户界面&#xff0c;操作直观便捷&#xff0c;并且兼容 W…

验证码的作用,为什么要存在验证码?

背景 在现代网络应用中&#xff0c;验证码被广泛使用以实现人机识别和减轻服务器负担。常见的验证码为以下几类&#xff1a; 图形验证码&#xff1a;通过展示一个随机生成的图形&#xff0c;要求用户输入对应的文字或数字来判断用户是否为真实用户。滑块验证码&#xff1a;用…

基于VS2022+Qt5+C++的网络调试助手开发

目录 一、前言 二、环境准备以及项目创建 三、 项目实现 1.ui界面设计 2.添加NetWork模块 QTcpSocket 和 QTcpServer QUdpSocket 3.主要功能实现 ①IP扫描 ②端口设置 ③数据接收 ④数据发送 ⑤日志保存 4.打包成exe 四、效果展示 五、总结 一、前言 我之前用…