代码随想录算法训练营第四十三天 | 1049. 最后一块石头的重量 II、494. 目标和、474.一和零

本文主要是介绍代码随想录算法训练营第四十三天 | 1049. 最后一块石头的重量 II、494. 目标和、474.一和零,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

01背包问题总结

今天总结一下昨天的分割等和子集和今天三道题的的思路,都属于01背包问题。
其实仔细观察就会发现,这四道题目本质上都是题目提供了一个集合,都需要在集合中选出满足目标的子集,且每个元素只能选一次,所以都是01背包问题。分割等和子集:不需要思路转换;石头:为了得到最小的石头,需要石头相撞,求出一个需要凑的子集,其和尽量接近整个集合求和的一半;目标和:加和减分别是子集,所以也是求子集,满足求和等于目标值;一和零:选字符串集合,尽量多个字符串。

然后具体解题过程中需要考虑pd数组的含义,其实也是对应到01背包问题模型中,物品是什么,重量是什么,价值是什么,目标又是什么(这一点直接决定dp数组及下标含义,和题目要求是直接相关的,例如,分割等和子集:dp[i]就是空间为i的背包能装的最大的元素和;石头:dp[i]是空间为i的背包能装的最大的重量;目标和:dp[i]是空间为i的背包能装的最大的元素和,以上三种提供的集合中都是数字,所以物品重量就是数值,价值也是数值;一和零:物品是字符串,重量是01个数,价值是字符串数量)

第一个要注意的点: 通过一和零问题会发现,背包可以是高维度的,这个和约束有关,如果约束只是物品重量,那就是一维的背包,如果是一和零这种问题,明显限制条件是0和1的个数,那么自然就是二维背包问题。 ==这里需要说明一下,==我这里讲的一维和二维问题不同于前面博客中讲解提到的一维二维数组的解法,前面的讲解中的二维指的是解决一维背包问题用二维数组。

接下来是主要收获的一点,一维背包问题可以用二维数组解决,好处在于容易理解,但是浪费空间,所以我们都用最低维度的数组解决相应的问题,但这样做就一定要注意遍历dp数组的顺序,为了防止物品被重复考虑加入,遍历都是从后向前遍历,无论一维还是二维,这里不太容易理解。我的理解方式是考虑一下高一个维度的数组就好了,比如解决一维背包问题,我用二维数组,虽然每行是从前向后遍历,但是填写当前行的dp数组利用的都是前面的行的信息,而用一维数组解决问题时,信息是会被覆盖的,为了利用原本的“前面行”的信息,所以从后向前遍历。

总结一下,01背包问题解决的都是在一定约束下找满足条件的子集的组合方式的问题(每个元素只能用一次),约束条件决定了问题的维度,解决问题用相应维度的数组,注意遍历方式要从后向前遍历。

题目链接:1049. 最后一块石头的重量 II

文章讲解:代码随想录 1049. 最后一块石头的重量 II讲解

视频讲解:这个背包最多能装多少?LeetCode:1049.最后一块石头的重量II

思路和解法

题目:
有一堆石头,用整数数组 stones 表示。其中 stones[i] 表示第 i 块石头的重量。

每一回合,从中选出任意两块石头,然后将它们一起粉碎。假设石头的重量分别为 x 和 y,且 x <= y。那么粉碎的可能结果如下:

如果 x == y,那么两块石头都会被完全粉碎;
如果 x != y,那么重量为 x 的石头将会完全粉碎,而重量为 y 的石头新重量为 y-x。
最后,最多只会剩下一块 石头。返回此石头 最小的可能重量 。如果没有石头剩下,就返回 0。
想法:
要理解石头相撞这个思路,本质上就是一个组合问题。

class Solution {
public:
//核心思路:尽量凑出两个求和相同的子集,这样对撞粉碎才能剩下最小的int lastStoneWeightII(vector<int>& stones) {int sum = 0;for (int i = 0; i < stones.size(); i++) {sum += stones[i];}int target = sum / 2;vector<int> dp(target + 1, 0);for (int i = 0; i < stones.size(); i++) {for (int j = target; j >= stones[i]; j--) {dp[j] = max(dp[j], dp[j - stones[i]] + stones[i]);}}return sum - dp[target] - dp[target];}
};

题目链接:494. 目标和

文章讲解:代码随想录 494. 目标和讲解

视频讲解:装满背包有多少种方法?| LeetCode:494.目标和

思路和解法

题目:
给你一个非负整数数组 nums 和一个整数 target 。

向数组中的每个整数前添加 ‘+’ 或 ‘-’ ,然后串联起所有整数,可以构造一个 表达式 :

例如,nums = [2, 1] ,可以在 2 之前添加 ‘+’ ,在 1 之前添加 ‘-’ ,然后串联起来得到表达式 “+2-1” 。
返回可以通过上述方法构造的、运算结果等于 target 的不同 表达式 的数目。
想法:
记住本质上还是组合选出满足目标的子集问题。

class Solution {
public:
//核心思路:本质上将数组分成了两个子集,一个加,一个减 需要通过理论推导得出加子集bagSize = sum + target
//然后转化为01背包问题 但是dp数组的含义变为:dp[i]:容量为i的背包恰好装满有多少种装法int findTargetSumWays(vector<int>& nums, int target) {int sum = 0;for (int i = 0; i < nums.size(); i++) {sum += nums[i];}//题目没有提及得不到target的情况,这里代码还是考虑了一下if (abs(target) > sum) {return 0;}if ((sum + target) % 2 == 1) {return 0;}int bagSize = (sum + target) / 2;vector<int> dp(bagSize + 1, 0);dp[0] = 1;for (int i = 0; i < nums.size(); i++) {for (int j = bagSize; j >= nums[i]; j--) {//理解:想象外层循环在遍历数字,内层循环在遍历背包,能搞好填满当前容量背包只有把当前数字装进去一种方法//dp[0]初始化为1的原因也就是第一个数字刚好填满相应容量背包时,算是一种方法dp[j] += dp[j - nums[i]];}}return dp[bagSize];}
};

题目链接:474.一和零

文章讲解:代码随想录 474.一和零讲解

视频讲解:装满这个背包最多用多少个物品?| LeetCode:474.一和零

思路和解法

题目:
给你一个二进制字符串数组 strs 和两个整数 m 和 n 。

请你找出并返回 strs 的最大子集的长度,该子集中 最多 有 m 个 0 和 n 个 1 。

如果 x 的所有元素也是 y 的元素,集合 x 是集合 y 的 子集 。
想法:
记住本质上还是组合选出满足目标的子集问题。

class Solution {
public:
//核心思路:m n是背包维度 物品就是字符串 字符串中所含的0和1的个数就是物品的重量 
//dp[i][j]:空间为i j的背包最多可以装多少个字符串int findMaxForm(vector<string>& strs, int m, int n) {vector<vector<int>> dp(m + 1, vector<int>(n + 1, 0));//最外层循环遍历物品for (string str : strs) {//需要先获取物品信息 包含几个0 几个1int x = 0;int y = 0;for (char c : str) {if (c == '0') {x++;} else {y++;}}//遍历背包 因为本身就是二维01背包 这里用的又是二维数组 所以两个维度都要从后向前遍历 防止一个物品多次加入for (int i = m; i >= x; i--) {for (int j = n; j >= y; j--) {//一种是考虑带上当前遍历的物品 另一种是考虑不带当前物品 选装的物品更多的方案留下dp[i][j] = max(dp[i - x][j - y] + 1, dp[i][j]);}}}return dp[m][n];}
};

这篇关于代码随想录算法训练营第四十三天 | 1049. 最后一块石头的重量 II、494. 目标和、474.一和零的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码

《Java中Date、LocalDate、LocalDateTime、LocalTime、时间戳之间的相互转换代码》:本文主要介绍Java中日期时间转换的多种方法,包括将Date转换为LocalD... 目录一、Date转LocalDateTime二、Date转LocalDate三、LocalDateTim

jupyter代码块没有运行图标的解决方案

《jupyter代码块没有运行图标的解决方案》:本文主要介绍jupyter代码块没有运行图标的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录jupyter代码块没有运行图标的解决1.找到Jupyter notebook的系统配置文件2.这时候一般会搜索到

Python通过模块化开发优化代码的技巧分享

《Python通过模块化开发优化代码的技巧分享》模块化开发就是把代码拆成一个个“零件”,该封装封装,该拆分拆分,下面小编就来和大家简单聊聊python如何用模块化开发进行代码优化吧... 目录什么是模块化开发如何拆分代码改进版:拆分成模块让模块更强大:使用 __init__.py你一定会遇到的问题模www.