代码随想录算法训练营Day41 | 0-1背包理论基础、416.分割等和子集

本文主要是介绍代码随想录算法训练营Day41 | 0-1背包理论基础、416.分割等和子集,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

0-1背包理论基础

基础

DP数组与其下标的含义

dp[i][j],i为物品编号,j为背包容量

dp[i][j]表示从下标为[0-i]的物品里任意取,放进容量为j的背包,价值总和最大是多少。

递推公式

分类:是否要放入下标为i的物品:

· 不放时最大价值为:dp[i - 1][j]

· 放入时最大价值为:dp[i - 1][j – weight[i]] + value[i]

递推取两者较大值:dp[i][j] = max(dp[i - 1][j], dp[i - 1][j – weight[i]] + value[i])

DP数组初始化

dp[i][j]由其上方格子和左上方范围内某一个格子初始化而来,所以需要初始化最上的一行最左的一列

        i = 0时,对于j < weight[0]的格子,初始化为0,往后的格子初始化为value[0]

        j = 0时,背包容量为0,装不下任何物品,所以最左列全部初始化为0

遍历顺序

先遍历物品(i)或先遍历背包(j)都可以,都能将dp数组填满

滚动数组优化

因为dp[i][j]的值只由i-1行元素推出,所以dp数组可以使用一维滚动数组来代替二维数组

注意使用滚动数组时不能先遍历背包,只能先遍历物品。遍历物品时遍历背包的顺序应该从右到左(思考一下覆盖的顺序)


416.分割等和子集

(这题其实没有提示挺难想到背包解法的,告诉我是背包题也想了半天)

这题用背包解得想明白 j 是什么:寻找两个总和相等的子集,等价于寻找一个和为所以数总和一半的子集,所以 j 是数的总和,而 j 的最大值应该是数组中所有数总和的一半

1、DP数组定义:一维数组,使用滚动数组来实现背包。方便理解使用二维数组来解释定义:dp[i][j]表示 n = i 时,数组下标[0, i]中取任意数所能得到的最大值,这个最大值不能超过j

        · weight[i] 和 value[i] 都等于 nums[i]

        · value[i] == nums[i] 使在遍历物品时不断取到最大值

        · weight[i] == nums[i] 使在遍历背包时最大值不超过总和的一半

        · 最后遍历完了所有物品和背包后,如果dp[-1][-1] == 总和的一半,说明能恰好取到一个子集,其总和为所有数总和的一半

2、DP数组初始化:i < nums[0]的格子,初始化为0,往后的格子初始化为value[0]

3、递推公式:常规0-1背包问题的递推公式(滚动数组实现):
        dp[j] = max(dp[j], dp[j - nums[i]] + nums[i]);

bool canPartition(vector<int>& nums) {int sum = 0;for (int n : nums)sum += n;if (sum % 2 == 1)return false;// weight[i]与value[i]都设置为nums[i]// 当背包大小为sum / 2时,看最大数总和是否也是sum / 2sum /= 2;vector<int> dp(sum + 1, 0);for (int j = 0; j < sum + 1; ++j)if (j >= nums[0]) dp[j] = nums[0];for (int i = 1; i < nums.size(); ++i) {for (int j = sum; j >= nums[i]; --j) {dp[j] = std::max(dp[j], dp[j - nums[i]] + nums[i]);}}return dp[sum] == sum;
}

这篇关于代码随想录算法训练营Day41 | 0-1背包理论基础、416.分割等和子集的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

Java集合之Iterator迭代器实现代码解析

《Java集合之Iterator迭代器实现代码解析》迭代器Iterator是Java集合框架中的一个核心接口,位于java.util包下,它定义了一种标准的元素访问机制,为各种集合类型提供了一种统一的... 目录一、什么是Iterator二、Iterator的核心方法三、基本使用示例四、Iterator的工

Java 线程池+分布式实现代码

《Java线程池+分布式实现代码》在Java开发中,池通过预先创建并管理一定数量的资源,避免频繁创建和销毁资源带来的性能开销,从而提高系统效率,:本文主要介绍Java线程池+分布式实现代码,需要... 目录1. 线程池1.1 自定义线程池实现1.1.1 线程池核心1.1.2 代码示例1.2 总结流程2. J

Spring的基础事务注解@Transactional作用解读

《Spring的基础事务注解@Transactional作用解读》文章介绍了Spring框架中的事务管理,核心注解@Transactional用于声明事务,支持传播机制、隔离级别等配置,结合@Tran... 目录一、事务管理基础1.1 Spring事务的核心注解1.2 注解属性详解1.3 实现原理二、事务事

JS纯前端实现浏览器语音播报、朗读功能的完整代码

《JS纯前端实现浏览器语音播报、朗读功能的完整代码》在现代互联网的发展中,语音技术正逐渐成为改变用户体验的重要一环,下面:本文主要介绍JS纯前端实现浏览器语音播报、朗读功能的相关资料,文中通过代码... 目录一、朗读单条文本:① 语音自选参数,按钮控制语音:② 效果图:二、朗读多条文本:① 语音有默认值:②

Vue实现路由守卫的示例代码

《Vue实现路由守卫的示例代码》Vue路由守卫是控制页面导航的钩子函数,主要用于鉴权、数据预加载等场景,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着... 目录一、概念二、类型三、实战一、概念路由守卫(Navigation Guards)本质上就是 在路

uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)

《uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)》在uni-app开发中,文件上传和图片处理是很常见的需求,但也经常会遇到各种问题,下面:本文主要介绍uni-app小程序项目中实... 目录方式一:使用<canvas>实现图片压缩(推荐,兼容性好)示例代码(小程序平台):方式二:使用uni

JAVA实现Token自动续期机制的示例代码

《JAVA实现Token自动续期机制的示例代码》本文主要介绍了JAVA实现Token自动续期机制的示例代码,通过动态调整会话生命周期平衡安全性与用户体验,解决固定有效期Token带来的风险与不便,感兴... 目录1. 固定有效期Token的内在局限性2. 自动续期机制:兼顾安全与体验的解决方案3. 总结PS

C#中通过Response.Headers设置自定义参数的代码示例

《C#中通过Response.Headers设置自定义参数的代码示例》:本文主要介绍C#中通过Response.Headers设置自定义响应头的方法,涵盖基础添加、安全校验、生产实践及调试技巧,强... 目录一、基础设置方法1. 直接添加自定义头2. 批量设置模式二、高级配置技巧1. 安全校验机制2. 类型