• 动态规划课堂2-----路径问题


    目录

    引言:

    例题1:不同路径

    例题2:不同路径II

    例题3:礼物的最⼤价值

    例题4:下降路径最⼩和

    例题5:最小路径和

    结语:


    引言:

    在学习完动态规划斐波那契数列模型后,相信大家对动态规划已经有了一定的了解,下面我们继续深入学习动态规划的路径问题,我们一般的解题步骤还是1. 状态表示,2.状态转移方程,3.初始化,4.填表顺序,5.返回值。在写代码时一定要把这5步考虑清楚再写代码,写代码的步骤为1.创建 dp表2.初始化3.填表4.返回值。

    例题1:不同路径

    链接:不同路径

    题目简介:

    一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为 “Start” )。

    机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为 “Finish” )。

    问总共有多少条不同的路径?

    解法(动态规划):

    1. 状态表示:

    对于这种「路径类」的问题,我们的状态表⽰⼀般有两种形式:

    (1)从[i, j] 位置出发到终点有几种路径。

    (2)从起始位置出发,到达[i, j] 位置有几种路径。

    本题我们选择第二种,dp[i][j] 表示:⾛到[i, j] 位置处,⼀共有多少种⽅式。

    2.状态转移方程

    简单分析⼀下。如果dp[i][j] 表⽰到达[i, j] 位置的⽅法数,那么到达[i, j] 位置之 前的⼀⼩步,有两种情况:

    (1)从[i, j] 位置的上⽅( [i - 1, j] 的位置)向下⾛⼀步,转移到[i, j] 位置

    (2)从[i, j] 位置的左⽅( [i, j - 1] 的位置)向右⾛⼀步,转移到[i, j] 位置。

    由于我们要求的是有多少种⽅法,因此状态转移⽅程就呼之欲出了: dp[i][j] = dp[i - 1] [j] + dp[i][j - 1] 。

    3.初始化

    使用辅助结点 。

    注意点:(1)辅助结点⾥⾯的值要「保证后续填表是正确的」(2)下标的映射关系。在本题中,「添加⼀⾏」,并且「添加⼀列」后,只需将dp[0][1] 的位置初始化为1 即可。

    4.填表顺序

    根据「状态转移⽅程」的推导来看,填表的顺序就是「从上往下」填每⼀⾏,在填写每⼀⾏的时候 「从左往右」。

    5.返回值

    根据「状态表⽰」,我们要返回dp[m][n](添加了一个辅助节点) 的值 。

    代码如下:

    动态规划编写代码的步骤比较固定,上面那5步是想好思路,下面这4步是编写代码的步骤。

    1. class Solution {
    2. public int uniquePaths(int m, int n) {
    3. //1.创建dp表
    4. //2.初始化
    5. //3.填表
    6. //4.返回值
    7. int[][] dp = new int[m + 1][n + 1];
    8. dp[0][1] = 1;
    9. for(int i = 1;i <= m;i++){
    10. for(int j = 1;j <= n;j++){
    11. dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
    12. }
    13. }
    14. return dp[m][n];
    15. }
    16. }

    时间复杂度:O(n * m)

    空间复杂度:O(n * m)

    例题2:不同路径II

    链接:不同路径II

    题目简介:

    一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为 “Start” )。

    机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为 “Finish”)。

    现在考虑网格中有障碍物。那么从左上角到右下角将会有多少条不同的路径?

    网格中的障碍物和空位置分别用 1 和 0 来表示。

     解法(动态规划):

    本题是例题一的进阶版,但是两题的解题步骤是类似的不同的地方在于状态转移方程。[i - 1, j] 与[i, j - 1] 位置都是可能有障碍的,此时从上⾯或者左边是不可能到达[i, j] 位置的,也就是说,此时的⽅法数应该是0。由此我们可以得出⼀个结论,只要这个位置上「有障碍物」,那么我们就不需要计算这个位置上的值,直接让它等于0即可。

    代码如下:

    1. class Solution {
    2. public int uniquePathsWithObstacles(int[][] obstacleGrid) {
    3. //1.创建dp表
    4. //2.初始化
    5. //3.填表
    6. //4.返回值
    7. int m = obstacleGrid.length;
    8. int n = obstacleGrid[0].length;
    9. int[][] dp = new int[m + 1][n + 1];
    10. dp[0][1] = 1;
    11. for(int i = 1;i <= m;i++){
    12. for(int j = 1;j <= n;j++){
    13. if(obstacleGrid[i - 1][j - 1] == 1){
    14. dp[i][j] = 0;
    15. }else{
    16. dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
    17. }
    18. }
    19. }
    20. return dp[m][n];
    21. }
    22. }

    时间复杂度:O(n * m)

    空间复杂度:O(n * m)

    例题3:礼物的最⼤价值

    链接:礼物的最⼤价值

    题目简介:

    现有一个记作二维矩阵 frame 的珠宝架,其中 frame[i][j] 为该位置珠宝的价值。拿取珠宝的规则为:

    • 只能从架子的左上角开始拿珠宝
    • 每次可以移动到右侧或下侧的相邻位置
    • 到达珠宝架子的右下角时,停止拿取

    注意:珠宝的价值都是大于 0 的。除非这个架子上没有任何珠宝,比如 frame = [[0]]

    解法(动态规划):

    1. 状态表示:

    dp[i][j] 表⽰:⾛到[i, j] 位置处,此时的最⼤价值。

    2.状态转移方程

    对于dp[i][j] ,我们发现想要到达[i, j] 位置,有两种⽅式:

    (1)从[i, j] 位置的上⽅[i - 1, j] 位置,向下⾛⼀步,此时到达[i, j] 位置能 拿到的礼物价值为dp[i - 1][j] + grid[i][j] .

    (2)从[i, j] 位置的左边[i, j - 1] 位置,向右⾛⼀步,此时到达[i, j] 位置能 拿到的礼物价值为dp[i][j - 1] + grid[i][j].

    最终dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]) + grid[i][j] 。

    3.初始化

    添加辅助节点

    4.填表顺序

    填表的顺序是「从上往下填写每⼀⾏」,「每⼀⾏从左往右」。

    5.返回值

    返回dp[m][n]的值

    代码如下:

    1. class Solution {
    2. public int jewelleryValue(int[][] frame) {
    3. //1.创建dp表
    4. //2.初始化
    5. //3.填表
    6. //4.返回值
    7. int m = frame.length;
    8. int n = frame[0].length;
    9. int[][] dp = new int[m + 1][n + 1];
    10. for(int i = 1;i <= m;i++){
    11. for(int j = 1;j <=n;j++){
    12. dp[i][j] = Math.max(dp[i - 1][j],dp[i][j - 1]) + frame[i - 1][j - 1];
    13. }
    14. }
    15. return dp[m][n];
    16. }
    17. }

    时间复杂度:O(n * m)

    空间复杂度:O(n * m)

    例题4:下降路径最⼩和

    链接:下降路径最⼩和

    题目简介:

    给你一个 n x n 的 方形 整数数组 matrix ,请你找出并返回通过 matrix 的下降路径  最小和 。

    下降路径 可以从第一行中的任何元素开始,并从每一行中选择一个元素。在下一行选择的元素和当前行所选元素最多相隔一列(即位于正下方或者沿对角线向左或者向右的第一个元素)。具体来说,位置 (row, col) 的下一个元素应当是 (row + 1, col - 1)(row + 1, col) 或者 (row + 1, col + 1) 。

    解法(动态规划):

    1. 状态表示:

    dp[i][j] 表⽰:到达[i, j] 位置时,所有下降路径中的最⼩和。

    2.状态转移方程

    对于普遍位置[i, j] ,根据题意得,到达[i, j] 位置可能有三种情况:

    (1)从正上⽅[i - 1, j] 位置转移到[i, j] 位置.

    (2)从左上⽅[i - 1, j - 1] 位置转移到[i, j] 位置;

      (3)   从右上⽅[i - 1, j + 1] 位置转移到[i, j] 位置;

    由于我们要的是最小的于是 dp[i][j] = min(dp[i - 1][j], min(dp[i - 1][j - 1], dp[i - 1][j + 1])) + matrix[i][j] 。

    3.初始化

    辅助节点

    在本题中,需要「加上⼀⾏」,并且「加上两列」。所有的位置都初始化为⽆穷⼤,然后将第⼀⾏ 初始化为0即可。

    4.填表顺序

    表的顺序是从上往下。

    5.返回值

    返回dp表中最后⼀⾏的最⼩值。

    代码如下:

    1. class Solution {
    2. public int minFallingPathSum(int[][] matrix) {
    3. //1.创建dp表
    4. //2.初始化
    5. //3.填表
    6. //4.返回值
    7. int m = matrix.length;//y
    8. int n = matrix[0].length;//x
    9. int[][] dp = new int[m + 1][n + 2];
    10. //辅助节点
    11. for(int i = 0;i <= m;i++){
    12. dp[i][0] = Integer.MAX_VALUE;
    13. dp[i][n + 1] = Integer.MAX_VALUE;
    14. }
    15. for(int i = 1;i <= n;i++){
    16. dp[m][i] = Integer.MAX_VALUE;
    17. }
    18. for(int i = m - 1;i >= 0;i--){
    19. for(int j = 1;j <= n;j++){
    20. int min = Math.min(Math.min(dp[i + 1][j - 1],dp[i + 1][j]),dp[i + 1][j + 1]);
    21. if(min == Integer.MAX_VALUE){
    22. min = 0;
    23. }
    24. dp[i][j] = matrix[i][j - 1] + min;
    25. }
    26. }
    27. int key = dp[0][1];
    28. for(int i = 1;i <= n;i++){
    29. if(key > dp[0][i]){
    30. key = dp[0][i];
    31. }
    32. }
    33. return key;
    34. }
    35. }

    时间复杂度:O(n * m)

    空间复杂度:O(n * m)

    例题5:最小路径和

    链接:最⼩路径和

    题目简介:

    给定一个包含非负整数的 m x n 网格 grid ,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。

    说明:每次只能向下或者向右移动一步。

    解法(动态规划):

    1. 状态表示:

    dp[i][j] 表⽰:到达[i, j] 位置处,最⼩路径和是多少。

    2.状态转移方程

    简单分析⼀下。如果dp[i][j] 表⽰到达到达[i, j] 位置处的最⼩路径和,那么到达 [i, j] 位置之前的⼀⼩步,有两种情况:

    (1)从[i - 1, j] 向下⾛⼀步,转移到[i, j] 位置;

    (2)从[i, j - 1] 向右⾛⼀步,转移到[i, j] 位置。

    由于到[i, j] 位置两种情况,并且我们要找的是最⼩路径,因此只需要这两种情况下的最⼩值,再加上 [i, j] 位置上本⾝的值即可。故最终结果为:dp[i][j] = min(dp[i - 1][j], dp[i][j - 1]) + grid[i][j]。

    3.初始化

    辅助节点

    在本题中,「添加⼀⾏」,并且「添加⼀列」后,所有位置的值可以初始化为⽆穷⼤,然后让 dp[0][1] = dp[1][0] = 0 即可。如下图所示:

    4.填表顺序

    「从上往下」填每⼀⾏,每⼀⾏「从左往右」。

    5.返回值

    返回的结果是dp[m][n] 。

    代码如下:

    1. class Solution {
    2. public int minPathSum(int[][] grid) {
    3. //1.创建dp表
    4. //2.初始化
    5. //3.填表
    6. //4.返回值
    7. int m = grid.length;//y
    8. int n = grid[0].length;//x
    9. int[][] dp = new int[m + 1][n + 1];
    10. for(int i = 2;i <= n;i++){
    11. dp[0][i] = Integer.MAX_VALUE;
    12. }
    13. for(int i = 2;i <= m;i++){
    14. dp[i][0] = Integer.MAX_VALUE;
    15. }
    16. for(int i = 1;i <= m;i++){
    17. for(int j = 1;j <= n;j++){
    18. dp[i][j] = Math.min(dp[i - 1][j],dp[i][j - 1]) + grid[i - 1][j - 1];
    19. }
    20. }
    21. return dp[m][n];
    22. }
    23. }

    时间复杂度:O(n * m)

    空间复杂度:O(n * m)

    结语:

    其实写博客不仅仅是为了教大家,同时这也有利于我巩固自己的知识点,和一个学习的总结,由于作者水平有限,对文章有任何问题的还请指出,接受大家的批评,让我改进,如果大家有所收获的话还请不要吝啬你们的点赞收藏和关注,这可以激励我写出更加优秀的文章。

  • 相关阅读:
    快速实战SQL - 检索数据
    知物由学 | 自监督学习助力内容风控效果提升
    关于冒泡排序算法的实验
    固定VMware中Linux系统的ip地址
    【深度学习 | 核心概念】那些深度学习路上必经的核心概念,确定不来看看? (五)
    能够注入Bean的XXXUtil工具类
    谷粒商城--整合SpringCloud Alibaba
    day05-离线留言和离线文件
    手把手教你自己搭建getwayWorker聊天服务器
    “绿色低碳+数字孪生“双轮驱动,解码油气管道站升级难点 | 图扑软件
  • 原文地址:https://blog.csdn.net/2301_80035594/article/details/136319398