代码随想录训练营Day 45|力扣1049. 最后一块石头的重量 II、494. 目标和、474.一和零

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

1.最后一块石头的重量2

视频讲解:动态规划之背包问题,这个背包最多能装多少?LeetCode:1049.最后一块石头的重量II_哔哩哔哩_bilibili

代码随想录

代码:

class Solution {
public:int lastStoneWeightII(vector<int>& stones) {int sum = 0;for(int i = 0;i < stones.size(); i++){sum += stones[i];}int begweight = sum / 2;// 初始化dp数组vector<int> dp(begweight + 1,0);// 递推公式for(int i = 0; i < stones.size(); i++){for(int j = begweight; j >= stones[i]; j--){dp[j] = max(dp[j],dp[j - stones[i]] + stones[i]);}}int result = sum - dp[begweight] - dp[begweight];return result;}
};

 思路:这道题和划分等和子集的题的思路很像,我们尽量把石头按照重量等分成两组,然后碰撞,这样剩下的石头重量最低。这道题也是把石头的重量看成是它的价值。

dp数组的含义:dp[j]表示 用物品1-i去装容量为j的背包所得的最大重量

dp数组的递推公式:就是一维滚动01背包的基础上,把价值换成了重量。 dp[j] = max(dp[j],dp[j - stones[i]] + stones[i])

dp数组的初始化:为了为了确保每个物品只被添加一次,我们在遍历物品的循环中使用了倒序遍历。这是因为如果在正序遍历的情况下,当我们计算 dp[j] 时,dp[j-weight[i]] 已经被更新过了,如果当前物品被多次添加,则 dp[j] 的值会受到影响,不再是我们期望的状态。

这道题我又错了,sos!!在写j的for循环的时候,把j的边界条件写错了,j >= stones[i]!!!!!

2.目标和

视频讲解:动态规划之背包问题,装满背包有多少种方法?| LeetCode:494.目标和_哔哩哔哩_bilibili

代码随想录

代码: 

class Solution {
public:int findTargetSumWays(vector<int>& nums, int target) {// 可以把数组分成left和right两个阵营。left是正数组。right是负数组。// 这样left+right=target left-right=sum// left=(target + sum)/2 只要去求用nums里的物品有多少种方法装满容量为left的背包就好了int sum = 0;for(int i = 0; i < nums.size(); i++){sum += nums[i];}if (abs(target) > sum) return 0; if((sum + target) % 2 == 1){ // 如果left都不是整数,说明没有求得target的表达式return 0;}int left = (sum + target)/2;// 定义和初始化dp数组vector<int> dp(left + 1,0);dp[0] = 1;// 递推公式for(int i = 0; i < nums.size(); i++){for(int j = left; j >= nums[i]; j--){dp[j] += dp[j - nums[i]]; // 可以看作是求什么情况下,再添一个元素就正好到容量j了(就像爬楼梯一样,只不过这道题没有限制只能爬一层台阶,所以我们出发的起点就很多}}return dp[left];}
};

 思路:

        这道题其实可以把数组分成left和right两个阵营。left是正数组。right是负数组。这样left+right=target left-right=sum推出left=(target + sum)/2。——>只要去求用nums里的物品有多少种方法装满容量为left的背包就好了。

        涉及到求方法类的问题,可以看作是求什么情况下,再添一个元素就正好到容量j了(就像爬楼梯一样,只不过这道题没有限制只能爬一层台阶,所以我们出发的起点就很多。也可以求用我们之前做过的路径总和来类比。

dp数组的含义:dp[j] 用物品0~i装满容量为j的背包有多少种方法

dp数组的递推公式:dp[j] += dp[j - nums[i] ]

dp数组的初始化:

        从递推公式可以看出,在初始化的时候dp[0] 一定要初始化为1,因为dp[0]是在公式中一切递推结果的起源,如果dp[0]是0的话,递推结果将都是0。

        这里有录友可能认为从dp数组定义来说 dp[0] 应该是0,也有录友认为dp[0]应该是1。

        其实不要硬去解释它的含义,咱就把 dp[0]的情况带入本题看看应该等于多少。

        如果数组[0] ,target = 0,那么 bagSize = (target + sum) / 2 = 0。 dp[0]也应该是1, 也就是说给数组里的元素 0 前面无论放加法还是减法,都是 1 种方法。

        所以本题我们应该初始化 dp[0] 为 1。

        可能有同学想了,那 如果是 数组[0,0,0,0,0] target = 0 呢。

        其实 此时最终的dp[0] = 32,也就是这五个零 子集的所有组合情况,但此dp[0]非彼dp[0],dp[0]能算出32,其基础是因为dp[0] = 1 累加起来的。

对了,这道题我一开始提交的时候显示我溢出了,后来加上了 if (abs(target) > sum) return 0;这句话就过了。

3.一和零

视频讲解:动态规划之背包问题,装满这个背包最多用多少个物品?| LeetCode:474.一和零_哔哩哔哩_bilibili

代码随想录

代码:

class Solution {
public:int findMaxForm(vector<string>& strs, int m, int n) {// 定义dp数组 这道题相当于从两个维度装背包,因此定义一个二维的dp数组vector<vector<int>> dp(m + 1,vector<int>(n + 1,0));// 递推公式for(string str : strs){ // 遍历物品int oneNum = 0;int zeroNum = 0;for(char c : str){ if(c == '0'){ // 统计每个物品里0、1的数量zeroNum++;}else{oneNum++;}}for(int j = m; j >= zeroNum; j--){for(int k = n; k >= oneNum; k--){dp[j][k] = max(dp[j][k],dp[j - zeroNum][k - oneNum] + 1);// 因为我们求的是装满容量为j,k的最多的物品数量,所以这里的value换成了数量1}}}return dp[m][n];}
};

 思路:

这道题相当于从两个维度(专门装0,和专门装1的背包容量)装背包,问装满背包的最多的物品数量为多少

dp数组的含义:dp[j][k]装满(专门装0容量为j,和专门装1容量为k)的背包,最多所用的物品数量

dp数组的递推公式:dp[j][k] = max(dp[j][k],dp[j - zeroNum][k - oneNum] + 1)

dp数组的初始化:为了不影响我们取最大值,全部初始化为0

dp数组的遍历顺序:就和滚动数组的一样,为了每个物品只被添加一次,倒叙遍历

这道题,我又又又错了,那个j和k的for循环和统计一个元素中的01个数的循环 是并列关系,我写成嵌套了[悲伤]

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



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

相关文章

利用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. 指

使用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.

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La