算法训练之动态规划(三)


♥♥♥~~~~~~欢迎光临知星小度博客空间~~~~~~♥♥♥

♥♥♥零星地变得优秀~也能拼凑出星河~♥♥♥

♥♥♥我们一起努力成为更好的自己~♥♥♥

♥♥♥如果这一篇博客对你有帮助~别忘了点赞分享哦~♥♥♥

♥♥♥如果有什么问题可以评论区留言或者私信我哦~♥♥♥

✨✨✨✨✨✨ 个人主页✨✨✨✨✨✨

        接下来这篇博客我们将继续在题目中使用巧妙的动态规划思想~准备好了吗~我们发车去探索奥秘啦~🚗🚗🚗🚗🚗🚗

下降路径的最小和

下降路径的最小和

这个题目看起来有点吓人,别怕,接下来我们一步步按照动态规划的思想来解决这道问题~

分析:

1、状态表示

        结合这里的题目要求+经验:我们这里的状态表示——dp[i][j]为到达该位置的最小路径和

2、状态转移方程

       我们以离【i】【j】位置最近的状态分析状态转移方程(题目要求最多相隔一列)所以有下面的三种情况:

1、到达该位置,可能是从第【i-1】【j】位置向下一步到达的

2、到达该位置,可能是从第【i-1】【j-1】位置向下再向右一步到达的

3、到达该位置,可能是从第【i-1】【j+1】位置向下再向左一步到达的

       所以到达该位置最小路径和也就是三种情况的最小值再加上当前位置的数据(注意下标映射关系),所以状态转移方程也就是:

                dp[i][j]=min(min(dp[i-1][j],dp[i-1][j-1]),dp[i-1][j+1]) + nums[i-1][j-1]

3、初始化

        我们可以看到,状态转移方程里面有i-1,j-1,j+1当i=0或者j=0或者j=n的时候显然会出现越界的情况,所以我们需要进行初始化

        结合前面如果不想初始化太麻烦,我们可以多申请一些空间,这里与前面不一样的是这里是我们这里还有一个j+1,那么我们这里也就可以多申请一行两列~那么怎么初始化这一行两列呢?

        我们结合例子来看看:

        我们首先来看看结合我们的逻辑dp[i][j]里面的值在具体的例子中是多少?

      我们可以看到第一行就是它本身的值,而第一行是受到我们增加的那一行影响的,所以我们增加的那一行应该全部初始化为0,才不会出错~接下来看后面的两行,都是在取上一行三个位置的最小值加上当前位置值得到的,那么旁边的两列如果也初始化为0的话,那么边界找到的最小值就是0了,这并不是数组里面的有效数据~所以为了避免影响,我们也就可以把两列位置设置成INT_MAX,题目给出数据大小范围,显然dp[i][j]是不会超过INT_MAX的,那么我们就可以这样进行初始化~

                第一行初始化为0,剩下的位置初始化为INT_MAX

       

 

        这种初始化方式还需要注意的是下标的映射关系~

4、填表顺序

        我们这里的逻辑是从上面依次推出下面的,所以填表顺序是从上向下

5、返回结果

      这里返回结果也与我们前面的不一样,因为它不是一个具体的位置,最小路径和存在于最后一行,所以我们可以找到dp表最后一行最小值进行返回~

代码实现:

class Solution 
{
public:int minFallingPathSum(vector<vector<int>>& matrix) {//1、创建dp表int m=matrix.size();int n=matrix[0].size();//多创建一行两列vector<vector<int>> dp(m+1,vector<int>(n+2,INT_MAX));//最开始就把所有位置初始化为INT_MAX//2、初始化//最开始就把所有位置初始化为INT_MAX了//初始化第一行为0for(int j=0;j<=n+1;j++){dp[0][j]=0;}//3、填表for(int i=1;i<=m;i++){for(int j=1;j<=n;j++){dp[i][j]=min(min(dp[i-1][j-1],dp[i-1][j]),dp[i-1][j+1])+matrix[i-1][j-1];//注意下标映射关系}}//4、返回结果//找到最后一行最小值进行返回int ret=dp[m][1];for(int i=2;i<=n;i++){ret=min(ret,dp[m][i]);}return ret;}
};

顺利通过~分析之后也就十分清楚了,我们来看看下一道题目~

最小路径和

最小路径和

我们同样可以一步步来分析:

1、状态表示

        结合这里的题目要求+经验:我们这里的状态表示——dp[i][j]为到达该位置的最小总和

2、状态转移方程

       我们以离【i】【j】位置最近的状态分析状态转移方程(题目要求只能向下或者向右移动)所以有下面的两种情况:

1、到达该位置,可能是从第【i-1】【j】位置向下一步到达的

2、到达该位置,可能是从第【i】【j-1】位置向右一步到达的

       所以到达该位置最小总和也就是两种情况的最小值再加上当前位置的数据(注意下标映射关系),所以状态转移方程也就是:

                dp[i][j]=min(dp[i-1][j],dp[i][j-1]) + nums[i-1][j-1]

3、初始化

        我们可以看到,状态转移方程里面有i-1,j-1当i=0或者j=0的时候显然会出现越界的情况,所以我们需要进行初始化

        结合前面如果不想初始化太麻烦,我们可以多申请一些空间,那么我们这里结合经验也就可以多申请一行一列~那么怎么初始化这一行一列呢?

        我们结合例子来看看:

        我们首先来看看结合我们的逻辑dp[i][j]里面的值在具体的例子中是多少?

      我们可以看到除了左上角的数就是它本身的值,其他的值都发生了变化,而左上角的数是受到它上面的位置和左边的位置影响的,结合状态转移方程  dp[i][j]=min(dp[i-1][j],dp[i][j-1]) + nums[i-1][j-1],所以它上面的位置和左边的位置应该初始化为0,才不会出错~接下来看剩下的位置,每一个位置都是在取当前位置上面的位置和左边的位置的最小值加上当前位置值得到的,那么如果也初始化为0的话,那么边界找到的最小值就是0了,这并不是数组里面的有效数据~所以为了避免影响,我们也就可以把剩下的设置成INT_MAX,题目给出数据大小范围,显然dp[i][j]是不会超过INT_MAX的,那么我们就可以这样进行初始化~

       dp[1][0]=dp[0][1]=0,剩下的位置初始化为INT_MAX

 

        这种初始化方式还需要注意的是下标的映射关系~

4、填表顺序

        我们这里的逻辑是从上面依次推出下面的,所以填表顺序是从上向下

5、返回结果

     右下角的位置就是我们想要的结果,直接返回dp[m][n]就可以了

有了前面题目的铺垫,相信这道题目就手到擒来了~

代码实现:

class Solution 
{
public:int minPathSum(vector<vector<int>>& grid) {//1、创建dp表int m=grid.size();int n=grid[0].size();//dp表最开始全部初始化为INT_MAXvector<vector<int>> dp(m+1,vector<int>(n+1,INT_MAX));//2、初始化dp[0][1]=dp[1][0]=0;//3、填表for(int i=1;i<=m;i++){for(int j=1;j<=n;j++){dp[i][j]=min(dp[i-1][j],dp[i][j-1])+grid[i-1][j-1];//注意下标映射关系}}//4、返回结果return dp[m][n];}
};

顺利通过~

地下城游戏

地下城游戏

一看这题目就有点吓人,我们同样按照我们之前的步骤来进行分析:

1、状态表示

        结合这里的题目要求+经验:状态表示一般有两种方式:

1、以某一个位置为结尾分析

        我们这里来看看如果我们以【i】【j】位置为结尾来分析的话,认为dp【i】【j】表示的是到该位置的最低健康点数,那么我们可以发现dp【i】【j】不仅仅受到前面位置的影响,还会受到后面位置的影响,这样就会十分麻烦,所以我们换一种方式~

2、以某一个位置为起点分析

        我们这里来看看如果我们以【i】【j】位置为起点来分析的话,认为dp【i】【j】表示的是从该位置开始到达指定位置的最低健康点数,那么dp【i】【j】只受到后面位置的影响,这样就更加方便~

2、状态转移方程

       我们以离【i】【j】位置最近的状态分析状态转移方程(题目要求只能向下或者向右移动)所以有下面的两种情况:

1、以该位置为起点,可能是向下一步到达第【i+1】【j】位置

2、以该位置为起点,可能是向右一步到达第【i】【j+1】位置

       所以dp【i】【j】也就是以该位置为起点,两种情况的最小值再减去当前位置的数据(这里增加不需要注意下标映射关系),所以状态转移方程也就是:

                dp[i][j]=min(dp[i+1][j],dp[i][j+1]) - nums[i][j]

        这样处理之后,还有一种情况就是dp[i][j]是负数了,也就是nums[i][j]很大,那么该位置的最小健康点数只需要是1经过加上nums[i][j]就可以了,所以还需要处理为负数的情况~

        dp[i][j]=max(1,dp[i][j])

3、初始化

        我们可以看到,状态转移方程里面有i+1,j+1当i=m-1或者j=n-1的时候显然会出现越界的情况,所以我们需要进行初始化

        结合前面如果不想初始化太麻烦,我们可以多申请一些空间,那么我们这里结合经验也就可以多申请一行一列~与前面不同的是这里是以【i】【j】位置为起点来分析,所以加的那一行一列在后面,那么怎么初始化这一行一列呢?

        我们结合例子来看看:

        我们首先来看看结合我们的逻辑dp[i][j]里面的值在具体的例子中是多少?

      我们可以看到除了右下角的数到达右边或者下边那一个位置至少还应该剩下1的健康点数,不能刚好到了就死亡了,所以dp【m-1】【n】=dp【m】【n-1】=1,接下来看剩下的位置,每一个位置都是在取当前位置下面的位置和右边的位置的最小值加上当前位置值得到的,那么如果也初始化为1的话,那么边界找到的最小值就是错误的~所以为了避免影响,我们也就可以把剩下的设置成INT_MAX,题目给出数据大小范围,显然dp[i][j]是不会超过INT_MAX的,那么我们就可以这样进行初始化~

       dp[m-1][n]=dp[m][n-1]=1,剩下的位置初始化为INT_MAX

 

        这种初始化方式还不需要注意下标的映射关系了~因为位置与数组一一对应的~

4、填表顺序

        我们这里的逻辑是从下面/右边依次推出上面/左边的,所以填表顺序是从下向上,从右向左

5、返回结果

     左上角的位置就是我们想要的结果,直接返回dp[0][0]就可以了

代码实现:

class Solution 
{
public:int calculateMinimumHP(vector<vector<int>>& d) {//1、创建dp表int m=d.size();int n=d[0].size();//最开始全部初始化为INT_MAXvector<vector<int>> dp(m+1,vector<int>(n+1,INT_MAX));//2、初始化dp[m-1][n]=dp[m][n-1]=1;//3、填表//注意填表顺序for(int i=m-1;i>=0;i--){for(int j=n-1;j>=0;j--){dp[i][j]=min(dp[i+1][j],dp[i][j+1])-d[i][j];//处理为负的情况dp[i][j]=max(1,dp[i][j]);}}//4、返回结果return dp[0][0];}
};

顺利通过~


♥♥♥本篇博客内容结束,期待与各位优秀程序员交流,有什么问题请私信♥♥♥

♥♥♥如果这一篇博客对你有帮助~别忘了点赞分享哦~♥♥♥

✨✨✨✨✨✨个人主页✨✨✨✨✨✨


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

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

相关文章

$_GET变量

$_GET 是一个超级全局变量&#xff0c;在 PHP 中用于收集通过 URL 查询字符串传递的参数。它是一个关联数组&#xff0c;包含了所有通过 HTTP GET 方法发送到当前脚本的变量。 预定义的 $_GET 变量用于收集来自 method"get" 的表单中的值。 从带有 GET 方法的表单发…

jQuery多库共存

在现代Web开发中&#xff0c;项目往往需要集成多种JavaScript库或框架来满足不同的功能需求。然而&#xff0c;当多个库同时使用时&#xff0c;可能会出现命名冲突、功能覆盖等问题。幸运的是&#xff0c;jQuery提供了一些机制来确保其可以与其他库和谐共存。本文将探讨如何实现…

MySQL 中的聚簇索引和非聚簇索引有什么区别?

MySQL 中的聚簇索引和非聚簇索引有什么区别&#xff1f; 1. 从不同存储引擎去考虑 在MySIAM存储引擎中&#xff0c;索引和数据是分开存储的&#xff0c;包括主键索引在内的所有索引都是“非聚簇”的&#xff0c;每个索引的叶子节点存储的是数据记录的物理地址&#xff08;指针…

Java从入门到“放弃”(精通)之旅——启航①

&#x1f31f;Java从入门到“放弃 ”精通之旅&#x1f680; 今天我将要带大家一起探索神奇的Java世界&#xff01;希望能帮助到同样初学Java的你~ (๑•̀ㅂ•́)و✧ &#x1f525; Java是什么&#xff1f;为什么这么火&#xff1f; Java不仅仅是一门编程语言&#xff0c;更…

三相电为什么没零线也能通电

要理解三相电为什么没零线也能通电&#xff0c;就要从发电的原理说起 1、弧形磁铁中加入电枢&#xff0c;旋转切割磁感线会产生电流 随着电枢旋转的角度变化&#xff0c;电枢垂直切割磁感线 电枢垂直切割磁感线&#xff0c;此时会产生最大电压 当转到与磁感线平行时&#xf…

文件上传做题记录

1&#xff0c;[SWPUCTF 2021 新生赛]easyupload2.0 直接上传php 再试一下phtml 用蚁剑连发现连不上 那就只要命令执行了 2&#xff0c;[SWPUCTF 2021 新生赛]easyupload1.0 当然&#xff0c;直接上传一个php是不行的 phtml也不行&#xff0c;看下是不是前端验证&#xff0c;…

【Pandas】pandas DataFrame head

Pandas2.2 DataFrame Indexing, iteration 方法描述DataFrame.head([n])用于返回 DataFrame 的前几行 pandas.DataFrame.head pandas.DataFrame.head 是一个方法&#xff0c;用于返回 DataFrame 的前几行。这个方法非常有用&#xff0c;特别是在需要快速查看 DataFrame 的前…

日语学习-日语知识点小记-构建基础-JLPT-N4阶段(1):承上启下,继续上路

日语学习-日语知识点小记-构建基础-JLPT-N4阶段(1):承上启下,继续上路 1、前言(1)情况说明(2)工程师的信仰2、知识点(1)普通形(ふつうけい)と思います(2)辞書形ことができます(3)Vたことがあります。(4)Vた とき & Vる とき3、单词(1)日语单词(2…

码率自适应(ABR)相关论文阅读简报

标题&#xff1a;Quality Enhanced Multimedia Content Delivery for Mobile Cloud with Deep Reinforcement Learning 作者&#xff1a;Muhammad Saleem , Yasir Saleem, H. M. Shahzad Asif, and M. Saleem Mian 单位: 巴基斯坦拉合尔54890工程技术大学计算机科学与工程系 …

汇编语言:指令详解

零、前置知识 1、数据类型修饰符 名称解释byte一个字节&#xff0c;8bitword单字&#xff0c;占2个字节&#xff0c;16bitdword双字&#xff0c;占4个字节&#xff0c;32bitqword四字&#xff0c;占8个字节&#xff0c;64bit 2、关键词解释 ptr&#xff1a;它代表 pointer&a…

蓝桥杯c ++笔记(含算法 贪心+动态规划+dp+进制转化+便利等)

蓝桥杯 #include <iostream> #include <vector> #include <algorithm> #include <string> using namespace std; //常使用的头文件动态规划 小蓝在黑板上连续写下从 11 到 20232023 之间所有的整数&#xff0c;得到了一个数字序列&#xff1a; S12345…

【C++算法】54.链表_合并 K 个升序链表

文章目录 题目链接&#xff1a;题目描述&#xff1a;解法C 算法代码&#xff1a; 题目链接&#xff1a; 23. 合并 K 个升序链表 题目描述&#xff1a; 解法 解法一&#xff1a;暴力解法 每个链表的平均长度为n&#xff0c;有k个链表&#xff0c;时间复杂度O(nk^2) 合并两个有序…

Java中的注解技术讲解

Java中的注解&#xff08;Annotation&#xff09;是一种在代码中嵌入元数据的机制&#xff0c;不直接参与业务逻辑&#xff0c;而是为编译器、开发工具以及运行时提供额外的信息和指导。下面我们将由浅入深地讲解Java注解的概念、实现原理、各种应用场景&#xff0c;并通过代码…

京东与喜茶关系破裂:切断所有合作 禁止进入办公场所

快科技4月10日消息&#xff0c;据报道&#xff0c;京东集团近日被曝出内部下发全员禁令&#xff0c;全面封杀喜茶产品进入办公区域。 据知情人士透露&#xff0c;京东人力行政部门发布的通知明确规定&#xff1a;全国各职场禁止与喜茶品牌开展任何形式的合作&#xff1b;员工不…

+++++背到厌倦。持续更新

Spring IoC 的工作流程: 读取 BeanDefinition: Spring 容器启动时&#xff0c;会读取 Bean 的配置信息 (例如 XML 配置文件、注解或 Java 代码)&#xff0c;并将这些配置信息转换为 BeanDefinition 对象。创建 Bean 实例: 根据 BeanDefinition 中的信息&#xff0c;Spring 容器…

如何在Git历史中抹掉中文信息并翻译成英文

如何在Git历史中抹掉中文信息并翻译成英文 在软件开发和版本控制领域&#xff0c;维护一个清晰、一致的代码历史记录是至关重要的。然而&#xff0c;有时我们可能会遇到需要修改历史提交的情况&#xff0c;比如删除敏感信息或修正错误。本文将详细探讨如何在Git历史中抹掉中文…

21 天 Python 计划:MySQL中DML与权限管理

文章目录 前言一、介绍二、MySQL数据操作&#xff1a;DML2.1 插入数据&#xff08;INSERT&#xff09;2.1.1 插入完整数据&#xff08;顺序插入&#xff09;2.1.2 指定字段插入数据2.1.3 插入多条记录2.1.4 插入查询结果 2.2 更新数据&#xff08;UPDATE&#xff09;2.3 删除数…

微信小程序 -- 原生封装table

文章目录 table.wxmltable.wxss注意 table.js注意 结果数据结构 最近菜鸟做微信小程序的一个查询功能&#xff0c;需要展示excel里面的数据&#xff0c;但是菜鸟找了一圈&#xff0c;也没发现什么组件库有table&#xff0c;毕竟手机端好像确实不太适合做table&#xff01; 菜鸟…

LangChain-输出解析器 (Output Parsers)

输出解析器是LangChain的重要组件&#xff0c;用于将语言模型的原始文本输出转换为结构化数据。本文档详细介绍了输出解析器的类型、功能和最佳实践。 概述 语言模型通常输出自然语言文本&#xff0c;但在应用开发中&#xff0c;我们经常需要将这些文本转换为结构化的数据格式…

【安全】加密算法原理与实战

为了理解SSL/TLS原理&#xff0c;大家需要掌握一些加密算法的基础知识。当然&#xff0c;这不是为了让大家成为密码学专家&#xff0c;所以只需对基础的加密算法有一些了解即可。基础的加密算法主要有哈希&#xff08;Hash&#xff0c;或称为散列&#xff09;​、对称加密(Symm…