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

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

1. 题目分析



题目链接选自力扣 : 下降路径最小和

496c95e15a3b4022a3f5276fe57db640.png


如果光看这个题目说明的话, 是有点抽象的. 我们结合实例 1 来看 :

0cd9122e384285c59c06b13872818356.png


总的来说就是, 起始点是第一行中的任意一点, 每个点只有三个方向可以走即向下, 左下, 右下. 当到达最后一行的任意一点即算作到达终点. 期间不同的路径上不同的点对应有不同的值, 最终那条路径的值总和最小则返回这个值.


2. 状态表示



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


3. 状态转移方程



利用我们之前的经验. 以最近的一步划分问题. 那么这个问题里最近的一步又是什么呢 ?

8c0f51a4c14d02ac5e2d63f209654151.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. 初始化



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

c0aef8508b0f08d06fa36e5ccc62c8db.png

还是根据我们之前新开一行一列的办法.但是这时候就不止多开一列了而是两列. 因为最后一列也需要初始化.


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


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


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


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


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

7ab244e257c7471eb017c0dd95991084.png

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


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;
    }
}


相关文章
|
消息中间件 存储 Java
RocketMQ(一):消息中间件缘起,一览整体架构及核心组件
【10月更文挑战第15天】本文介绍了消息中间件的基本概念和特点,重点解析了RocketMQ的整体架构和核心组件。消息中间件如RocketMQ、RabbitMQ、Kafka等,具备异步通信、持久化、削峰填谷、系统解耦等特点,适用于分布式系统。RocketMQ的架构包括NameServer、Broker、Producer、Consumer等组件,通过这些组件实现消息的生产、存储和消费。文章还提供了Spring Boot快速上手RocketMQ的示例代码,帮助读者快速入门。
|
关系型数据库 Linux 数据库
Linux系统之安装PostgreSQL数据库
Linux系统之安装PostgreSQL数据库
2510 1
|
Java 数据库连接 数据库
springboot项目运行时报错:HikariPool-1 - Exception during pool initialization.
springboot项目运行时报错:HikariPool-1 - Exception during pool initialization.
1324 0
|
5天前
|
云安全 人工智能 安全
AI被攻击怎么办?
阿里云提供 AI 全栈安全能力,其中对网络攻击的主动识别、智能阻断与快速响应构成其核心防线,依托原生安全防护为客户筑牢免疫屏障。
|
15天前
|
域名解析 人工智能
【实操攻略】手把手教学,免费领取.CN域名
即日起至2025年12月31日,购买万小智AI建站或云·企业官网,每单可免费领1个.CN域名首年!跟我了解领取攻略吧~
|
9天前
|
安全 Java Android开发
深度解析 Android 崩溃捕获原理及从崩溃到归因的闭环实践
崩溃堆栈全是 a.b.c?Native 错误查不到行号?本文详解 Android 崩溃采集全链路原理,教你如何把“天书”变“说明书”。RUM SDK 已支持一键接入。
598 215
|
4天前
|
编解码 Linux 数据安全/隐私保护
教程分享免费视频压缩软件,免费视频压缩,视频压缩免费,附压缩方法及学习教程
教程分享免费视频压缩软件,免费视频压缩,视频压缩免费,附压缩方法及学习教程
236 138