代码随想录算法训练营第四十七天(动态规划篇)| 416. 分割等和子集

本文主要是介绍代码随想录算法训练营第四十七天(动态规划篇)| 416. 分割等和子集,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

416. 分割等和子集

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

思路

回溯——超时

首先想到之前的回溯算法,寻找数组中加和等于sum(nums)/2的子集,但对于大数组超时了:

class Solution(object):def backtracking(self, nums, startIdx, curSum):if curSum > sum(nums)/2:return if curSum == sum(nums)/2 and sum(nums)%2 == 0:return Truefor i in range(startIdx, len(nums)):curSum += nums[i]if self.backtracking(nums, i+1, curSum) == True:return TruecurSum -= nums[i]return Falsedef canPartition(self, nums):return self.backtracking(nums, 0, 0)

动态规划

可以把数组中的每个元素看作一个物体,那么寻找符合要求的子集相当于把将所选的物体装入一定容量的背包。根据之前的学习,我们知道如何求定容背包能装的最大价值,如果把每一个元素的数值同时当成该物体的重量和价值,就可以用01背包求解。

如果容量为sum/2的背包能装的最大价值为sum/2(即可以被装满),则说明这个数组可以被分割成两个子集。

e,g, 对于数组[1, 2, 5], 有三个物体,它们的重量和价值分别为1,2,5,那对于容量为sum/2 = 4的背包,最多只能装1+2 = 3的价值,说明不能被分割。

1. dp数组定义

容量为j的背包,所背的物品价值最大可以为dp[j]。

若dp[j] == j 说明集合中的某子集总和正好可以凑成总和j;如果dp[sum/2] = sum/2,说明数组可以被分割成等和子集。

2. dp递推公式

01背包的递推公式为:dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);

本题,相当于背包里放入物品i的重量是nums[i],其价值也是nums[i]。

所以递推公式:dp[j] = max(dp[j], dp[j - nums[i]] + nums[i]);

3. 初始条件

当背包的容量为0时,能装的最大价值为0,所以dp[0]=0。如果题目给的价值都是正整数那么非0下标都初始化为0就可以了,如果题目给的价值有负数,那么非0下标就要初始化为负无穷。

4. 遍历顺序

根据01背包的滚动数组的方法:

代码随想录算法训练营第四十六天(动态规划篇)|01背包(滚动数组方法)-CSDN博客

如果使用一维dp数组,物品遍历的for循环放在外层,遍历背包容量的for循环放在内层,且内层for循环倒序遍历!

5. 举例推导dp数组

代码实现

class Solution(object):def canPartition(self, nums): # 剪枝:如果数组总和为奇数,则不能分割成等和子集if sum(nums)%2 == 1:return Falsetarget = sum(nums)/2dp = [0] * (target + 1)for num in nums:for j in range(target, num-1, -1): # 等于省略了之前if的条件之前的题需要trace weight[i]的值dp[j] = max(dp[j], dp[j-num] + num)return dp[target] == target

需要注意,在循环背包容量时,我们用

for j in range(target, num-1, -1):

因为对于重量(价值)为num的物体只能放在容量大于他自身重量的包里。这一个条件在之前的01背包代码是由下面的if语句实现:

if weight[i] > j:dp[j] = dp[j]

如果当前重量大于背包容量,就选择不装物品i,这相当于不考虑背包容量小于当前物体的值,即不在for循环中遍历。事实上之前的代码:

for i in range(objNum): # 遍历物体for j in range(bagWeight, 0, -1):  #遍历背包容量if weight[i] > j:dp[j] = dp[j]else:dp[j] = max(dp[j], dp[j - weight[i]] + value[i])

也可以改为:

for i in range(objNum): # 遍历物体for j in range(bagWeight, weight[i] - 1, -1):  # 遍历背包容量dp[j] = max(dp[j], dp[j - weight[i]] + value[i])

这篇关于代码随想录算法训练营第四十七天(动态规划篇)| 416. 分割等和子集的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

go动态限制并发数量的实现示例

《go动态限制并发数量的实现示例》本文主要介绍了Go并发控制方法,通过带缓冲通道和第三方库实现并发数量限制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录带有缓冲大小的通道使用第三方库其他控制并发的方法因为go从语言层面支持并发,所以面试百分百会问到

Python实现PDF按页分割的技术指南

《Python实现PDF按页分割的技术指南》PDF文件处理是日常工作中的常见需求,特别是当我们需要将大型PDF文档拆分为多个部分时,下面我们就来看看如何使用Python创建一个灵活的PDF分割工具吧... 目录需求分析技术方案工具选择安装依赖完整代码实现使用说明基本用法示例命令输出示例技术亮点实际应用场景扩

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java

一文详解SpringBoot中控制器的动态注册与卸载

《一文详解SpringBoot中控制器的动态注册与卸载》在项目开发中,通过动态注册和卸载控制器功能,可以根据业务场景和项目需要实现功能的动态增加、删除,提高系统的灵活性和可扩展性,下面我们就来看看Sp... 目录项目结构1. 创建 Spring Boot 启动类2. 创建一个测试控制器3. 创建动态控制器注

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN