动态规划之下降路径最小和

news/2024/11/17 3:32:58/

1. 题目分析

题目链接选自力扣 : 下降路径最小和
image.png
如果光看这个题目说明的话, 是有点抽象的. 我们结合实例 1 来看 :
image.png
总的来说就是, 起始点是第一行中的任意一点, 每个点只有三个方向可以走即向下, 左下, 右下. 当到达最后一行的任意一点即算作到达终点. 期间不同的路径上不同的点对应有不同的值, 最终那条路径的值总和最小则返回这个值.

2. 状态表示

我们以 dp[i][j] 表示从第一行的某个位置出发到达 ( i, j ) 位置时的最小路径和.

3. 状态转移方程

利用我们之前的经验. 以最近的一步划分问题. 那么这个问题里最近的一步又是什么呢 ?
image.png
很容易看出, 想要到达 ( i, j ) 位置一共有三种最近的情况.

  1. 从 ( i-1, j-1 ) 位置到达 ( i, j ) 位置

无论从第一行中的那个位置开始, 都需要先经过指定的 ( i-1, j-1 ) . 然后再从这个点到达 ( i, j ) 位置. 那么这种情况下,这条路径的最小和也就是到达 ( i-1, j-1 ) 位置的最小和, 正好对应我们的状态表示, 即 dp[i-1][j-1]. 最后在加上到达 ( i, j ) 位置的值 matrix[i][j].

  1. 从 ( i-1, j ) 位置到达 ( i, j ) 位置

同样的 从 ( i-1, j ) 到达 ( i, j ) 位置的最小和即为 dp[i-1][j] + matrix[i][j]

  1. 从 ( i-1, j+1 ) 位置到达 ( i, j ) 位置

同理, 从 ( i-1, j ) 到达 ( i, j ) 位置的最小和即为 dp[i-1][j+1] + matrix[i][j]

要找的是所有路径的最小和. 因此最终的 dp[i][j] = min(dp[i-1][j-1], dp[i-1][j], dp[i-1][j+1]) + matrix[i][j]

4. 初始化

初始化是为了保证填表的时候不会发生错误. 根据状态转移方程, 要填写某个点的时候, 需要知道它上一层的正下、左下、右下三个位置的值的. 第一行和第一列以及最后一列的值根据状态转移方程进行填写的时候都会发生越界错误. 因此这都是我们需要进行初始化的.
image.png
还是根据我们之前新开一行一列的办法.但是这时候就不止多开一列了而是两列. 因为最后一列也需要初始化.

这里的初始化是有很多细节的. 针对不同的位置初始化有所不同

1.当我们初始化第一行的第一个位置时.

它依赖上面的三个位置都是我们新增的位置. 不受其他具体值的影响. 因此这些新增的位置的值也不能影响这个位置原本的值 matrix[0][0. 根据状态转移方程我们取得是三个方向的最小值. 当这三个方向取 0 的时候, 最小值一定就是 matrix[0][0] 本身了. 因此新增的第一行初始化都为 0
image.png

  1. 对第二行的第一个位置进行初始化的时候

对这个位置进行填表的时候, 根据状态转移方程它受到三个位置的影响和自身的值. 但是右下方向红星的值是我们防止初始化新增的, 它不能影响最终填表的值, 这个位置的值只能受另外两个位置的直接影响. 状态转移方程中取得是三个位置的最小值. 因此新增的第一列初始化都为无穷大
在这里插入图片描述

同理, 最后一列的新增位置的初始化也应该为无穷大.

5. 填表顺序

从状态转移方程不难看出, 填写某个位置时需要知道上一层的三个方向的位置. 因此填表顺序是从上到下每一行. 每一行中从哪儿开始都行.只需要确保从上往下填写每一行就行.

6. 返回值

返回值这里有点特殊, 根据我们的状态表示, 是从第一行的任意一点到达结尾的任意一点的最小路径和. 因此在最后一行的每一个点都有可能是最小和. 因此返回的是最后一行值最小的那个

代码演示

class Solution {public int minFallingPathSum(int[][] matrix) {// 1. 建立 dp 表int m = matrix.length;int n = matrix[0].length;int[][] dp = new int[m + 1][n + 2];// 2. 初始化// 第一行初始化为 0// 第一列和最后一列初始化为无穷大for(int i = 1; i <= m; i++) {dp[i][0] = dp[i][n+1] = Integer.MAX_VALUE;}// 3. 填写 dp 表for(int i = 1; i <= m; i++) {for(int j = 1; j <= n; j++) {// 需要注意, 没有三个参数的最小值方法dp[i][j] = Math.min(dp[i-1][j-1], Math.min(dp[i-1][j], dp[i-1][j+1]))+ matrix[i-1][j-1];}}// 如果为默认为 0, 第一次比较 dp[m][j] 为正值时会影响最小取值int taget = Integer.MAX_VALUE; // 为最大值才不会影响最小值取值// 4. 确认返回值for(int j = 1; j <= n; j++) {taget = Math.min(taget, dp[m][j]);}return taget;}
}

http://www.ppmy.cn/news/578555.html

相关文章

技术问题epic无法启动 任务栏不断的闪 打不开

笔记本 从来没有装过epic 安装后打开一直在任务栏闪烁就是打不开重新下载 换低版本安装包和复制安装后的文件夹打开也是一样。

apk闪退_解决安卓手机闪退的通用办法!

原标题&#xff1a;解决安卓手机闪退的通用办法&#xff01; 丫丫网资讯&#xff0c;很多安卓用户都会反应手机在使用过程中会出现手机闪退问题&#xff0c;今天这个问题也许可以解决了。一起来瞅瞅吧&#xff01; 1、安卓手机由于其系统特性原因&#xff0c;在手机缓存过多的情…

王国纪元服务器不稳定,王国纪元闪退怎么解决 游戏闪退解决方法

王国纪元闪退怎么解决&#xff1f;最近玩家更新游戏之后&#xff0c;经常出现闪退的情况&#xff0c;在这里为你提供闪退解决方法&#xff1a; 游戏闪退解决方法 1、手机系统问题。对于安卓系统现在主流的系统都是4.0以上如果你手机的版本低于2.3.0&#xff0c;那么你该考虑升级…

避难所Android闪退,iOS/安卓版《辐射:避难所》Fallout Shelter攻略:闪退进不去解决办法...

辐射避难所Fallout Shelter闪退进不去解决办法&#xff1a; 1、网络异常 由于《辐射&#xff1a;避难所》(Fallout Shelter)是一款需要联网的游戏&#xff0c;所以如果我们的网络出现异常很容易造成辐射避难所闪退Fallout Shelter进不去的情况出现&#xff0c;如果网络连接异常…

行星边际2-planetSide2插件recursion tracker(00插件)の初体验

写在最前面&#xff1a; 整个使用过程中&#xff0c;注册账号这一步必须要科学上网&#xff0c;如果不能科学上网&#xff0c;务必先想办法获取到账号。 1、前往官网 官网 2、获取客户端 &#xff08;官网首页点击Get Client 或者点这里&#xff0c;下载不动点这里&#x…

win10玩武装突袭3一会就闪退的解决方法

很多玩家们在使用win10正式版系统玩武装突袭3&#xff08;arma3&#xff09;时&#xff0c;会出现游戏闪退、崩溃的情况&#xff0c;很是影响游戏的体验&#xff0c;我们要怎么解决这一情况呢&#xff1f;没关系&#xff0c;下面小编就来为大家分享关于win10玩武装突袭3一会就闪…

c++读取文件之---yaml-cpp使用

实际项目总会遇到有很多超参数的情况&#xff0c;用常规的结构体等无法有效的涵盖所有&#xff0c;为了方便用户进行配置使用&#xff0c;因此使用yaml的方式进行编辑配置&#xff0c;因此去调研使用了yaml-cpp的使用方法。 1、yaml-cpp下载和编译 下载方式很简单&#xff0c…

打不开Eclipse,Eclipse闪退(已解决)

不知道什么原因我的Eclipse打不开了&#xff0c;打开一半就闪退了&#xff0c;并生成了日志文件&#xff0c;在网上百度了一顿&#xff0c;没有解决办法&#xff0c;后来废了九牛二虎之力终于解决了: # # A fatal error has been detected by the Java Runtime Environment: # …