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

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

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数据库
2718 1
|
Java 数据库连接 数据库
springboot项目运行时报错:HikariPool-1 - Exception during pool initialization.
springboot项目运行时报错:HikariPool-1 - Exception during pool initialization.
1350 0
|
2天前
|
数据采集 人工智能 安全
|
11天前
|
云安全 监控 安全
|
3天前
|
自然语言处理 API
万相 Wan2.6 全新升级发布!人人都能当导演的时代来了
通义万相2.6全新升级,支持文生图、图生视频、文生视频,打造电影级创作体验。智能分镜、角色扮演、音画同步,让创意一键成片,大众也能轻松制作高质量短视频。
1007 151
|
3天前
|
编解码 人工智能 机器人
通义万相2.6,模型使用指南
智能分镜 | 多镜头叙事 | 支持15秒视频生成 | 高品质声音生成 | 多人稳定对话
|
16天前
|
机器学习/深度学习 人工智能 自然语言处理
Z-Image:冲击体验上限的下一代图像生成模型
通义实验室推出全新文生图模型Z-Image,以6B参数实现“快、稳、轻、准”突破。Turbo版本仅需8步亚秒级生成,支持16GB显存设备,中英双语理解与文字渲染尤为出色,真实感和美学表现媲美国际顶尖模型,被誉为“最值得关注的开源生图模型之一”。
1701 9