当前位置: 首页 > article >正文

LeetCode算法题解(动态规划)|LeetCoed62. 不同路径、LeetCode63. 不同路径 II

一、LeetCoed62. 不同路径

题目链接:62. 不同路径
题目描述:

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

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

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

示例 1:

输入:m = 3, n = 7
输出:28

示例 2:

输入:m = 3, n = 2
输出:3
解释:
从左上角开始,总共有 3 条路径可以到达右下角。
1. 向右 -> 向下 -> 向下
2. 向下 -> 向下 -> 向右
3. 向下 -> 向右 -> 向下

示例 3:

输入:m = 7, n = 3
输出:28

示例 4:

输入:m = 3, n = 3
输出:6

提示:

  • 1 <= m, n <= 100
  • 题目数据保证答案小于等于 2 * 109
算法分析:
dp数组及下标含义:

可以创建二维的dp[i][j]来表示到达坐标为(i,j)网格的路径数。

递推公式:

对于每一个网格(除最上面那一排和最左边那一列),到达它的方式有两种,一种是从上面那一个网格往下走一步,另一种是从左边那一个网格往右走一步,所以到达该网格的路径是上面那一个网格和左边那一个网格的路径之和,

dp[i][j] = dp[i-1][j]+dp[i][j-1];(i>0&&j>0)

初始化:

初始化最上边那一排和最左边那一列,到达那里的路径都是1。

遍历顺序:

因为每个网格的路径都是由它上边那一个和左边那一个网格决定的,所以从左到右从上到下遍历该二维网格,或者从上到下从左到右遍历都行。

如果结果不准确,请打印dp数组验证。

代码如下:

class Solution {
    public int uniquePaths(int m, int n) {
        int[][] dp = new int[m][n];
        //初始化,最上面那一排和最左边那一列的路径都只有一个
        for(int i = 0; i < n; i++)
        dp[0][i] = 1;
        for(int i = 1; i < m; i++)
        dp[i][0] = 1;
        for(int i = 1; i < m; i++) {
            for(int j = 1; j < n; j++) {
                dp[i][j] = dp[i - 1][j] + dp[i][j - 1];//到达右下角每个网格的路径为上面到达上面那一格网格和到达左边那一格网格的路径之和
            }
        }
        return dp[m - 1][n - 1];//返回到达最右下角网格的路径总数

    }
}

时间复杂度o(n*m),空间复杂度o(n*m);

二、LeetCode63. 不同路径 II

题目链接:63. 不同路径 II
题目描述:

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

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

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

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

示例 1:

输入:obstacleGrid = [[0,0,0],[0,1,0],[0,0,0]]
输出:2
解释:3x3 网格的正中间有一个障碍物。
从左上角到右下角一共有 2 条不同的路径:
1. 向右 -> 向右 -> 向下 -> 向下
2. 向下 -> 向下 -> 向右 -> 向右

示例 2:

输入:obstacleGrid = [[0,1],[0,0]]
输出:1

提示:

  • m == obstacleGrid.length
  • n == obstacleGrid[i].length
  • 1 <= m, n <= 100
  • obstacleGrid[i][j] 为 0 或 1
算法分析:
dp数组下标含义:

dp[i][j]表示到达坐标为(i,j)的网格的总路径。

初始化:

对于左边那一列,只能有从上往下一条路径,如果上方任意网格当中有障碍物,那么机器人就不可能到达障碍物下方,也就是说障碍物下方的路径为0;

       boolean flag = true;
        for(int i = 0; i < m; i++) {
            if(obstacleGrid[i][0] == 1) flag = false;
            if(flag) dp[i][0] = 1;
            else dp[i][0] = 0; 
        }

同理,对于最上方的那一行网格,只能有从左往右一条路径,如果左方任意网格当中有障碍物,那么机器人就不可能到达障碍物右方。

 flag = true;
        for(int i = 0; i < n; i++) {
            if(obstacleGrid[0][i] == 1) flag = false;
            if(flag) dp[0][i] = 1;
            else dp[0][i] = 0;
        }
递推公式:

如果当前网格有障碍物,那么机器人不可能当前网格,当前网格的路径为0;

如果没有障碍物,那么当前路径的数量就是上一个网格和左边那一个网格的路径之和。

遍历顺序:

从上到下,从左往右依次遍历。

如果结果有误,打印dp数组检查验证。

代码如下:

class Solution {
    public int uniquePathsWithObstacles(int[][] obstacleGrid) {
        int m = obstacleGrid.length;
        int n = obstacleGrid[0].length;
        int[][] dp = new int[m][n];
        boolean flag = true;
        for(int i = 0; i < m; i++) {//对于第一列上的网格只能有一条从上往下的路径,如果上方任意一个网格中有障碍物,那么机器人就不可能到达障碍物的下方
            if(obstacleGrid[i][0] == 1) flag = false;
            if(flag) dp[i][0] = 1;
            else dp[i][0] = 0; 
        }
        flag = true;
        for(int i = 0; i < n; i++) {//同理对于第一行上的网格只能有一条从左往右的路径,如果左方任意一个网格中有障碍物,那么机器人就不可能到达障碍物的右方。
            if(obstacleGrid[0][i] == 1) flag = false;
            if(flag) dp[0][i] = 1;
            else dp[0][i] = 0;
        }

        for(int i = 1; i < m; i++) {
            for(int j = 1; j < n; j++) {
                if(obstacleGrid[i][j] == 0) {//如果当前网格没有障碍物,那么到达它的路径就是上面那一个的网格和路左边那一个的网格路径之和
                    dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
                }else dp[i][j] = 0;//如果当前网格有障碍物,那么到达该网格的路径为0
            }
        }
        return dp[m - 1][n - 1];

    }
}

总结

二位dp数组有点难度,但只要掌握了递归五部曲不难。


http://www.kler.cn/a/136823.html

相关文章:

  • w~多模态~合集1
  • 嵌入式学习(21)-正点原子脱机下载器Mini-Pro的使用
  • k8s系列--docker拉取镜像导入k8s的containerd中
  • oceanbase集群访问异常问题处理
  • Docker--Docker Container(容器) 之 操作实例
  • 如何在IDEA一个窗口中导入多个项目
  • Go语言读取文件内容
  • 基于饥饿游戏算法优化概率神经网络PNN的分类预测 - 附代码
  • Threejs_08 纹理颜色的调整(颜色空间的设置)
  • 系列一、介绍
  • 【旅游行业】Axure旅游社交平台APP端原型图,攻略门票酒店民宿原型案例
  • 【经验分享】Ubuntu如何设置swap交换
  • 数据结构【DS】队列的应用
  • V8引擎隐藏类(VIP课程)
  • 2023亚太杯数学建模思路 - 案例:感知机原理剖析及实现
  • Web3 分布式存储 IPFS(Web3项目一实战之四)
  • 轻量封装WebGPU渲染系统示例<36>- 广告板(Billboard)(WGSL源码)
  • “伙伴计划·伙伴领航站”春晖团队在蟠龙社区开展青少年书香阅读陪伴活动
  • 动态顺序表
  • 科大讯飞 vue.js 语音听写流式实现 全网首发
  • 程序员有必要考个 985 非全日制研究生嘛?
  • Linux 时区设置
  • 信息系统项目管理师-范围管理论文提纲
  • house of husk
  • 通过汇编理解cortex-m3:第0章
  • .Net中Redis的Hash表操作