当前位置:   article > 正文

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

动态规划课堂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)

结语:

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

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小蓝xlanll/article/detail/179776?site
推荐阅读
相关标签
  

闽ICP备14008679号