一、需求
-
一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为“Start” ),机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为“Finish”),现在考虑网格中有障碍物。那么从左上角到右下角将会有多少条不同的路径?
- 网格中的障碍物和空位置分别用
1
和0
来表示。
输入:obstacleGrid = [[0,0,0],[0,1,0],[0,0,0]]
输出:2
解释:
3x3 网格的正中间有一个障碍物。
从左上角到右下角一共有 2 条不同的路径:
1. 向右 -> 向右 -> 向下 -> 向下
2. 向下 -> 向下 -> 向右 -> 向右
提示:
m == obstacleGrid.length
n == obstacleGrid[i].length
1 <= m, n <= 100
obstacleGrid[i][j] 为 0 或 1
二、动态规划
2.1 思路分析
- 根据题意求从(0,0)~(i,j)的所有有效路径,隐含的信息就是让你求方案总数,因此想到动态规划;
- 定义状态:f(i,j)表示(0,0)~(i,j)的有效路径;
- 转移方程:由于每次移动只能向右或向下,若(i,j)不是障碍,要想到达f(i,j),需要知道f(i-1,j)与f(i,j-1),它们的和就是f(i,j),如果是障碍,则f(i,j)为0;
2.2 代码实现
class Solution {
public int uniquePathsWithObstacles(int[][] obstacleGrid) {
int m = obstacleGrid.length;
int n = obstacleGrid[0].length;
int[][] f = new int[m][n];
//f[0][0] = obstacleGrid[0][0] == 0 ? 1 : 0;
//初始化第一列的各个状态
//&&运算符使得当有一个障碍时,从该障碍走的路径均为0
for(int i = 0; i < m && obstacleGrid[i][0] == 0; i++) {
f[i][0] = 1;
}
//初始化第一行的各个状态
for(int j = 0; j < n && obstacleGrid[0][j] == 0; j++) {
f[0][j] = 1;
}
for(int i = 1; i < m; i++) {
for(int j = 1; j < n; j++) {
if(obstacleGrid[i][j] == 0) {
f[i][j] = f[i - 1][j] + f[i][j - 1];
}
}
}
return f[m - 1][n - 1];
}
}
2.3 复杂度分析
- 时间复杂度为O(mn),其中m为二维数组的行数,n为二维数组的列数;
- 空间复杂度为O(mn);
三、学习地址
作者:sweetiee
链接:https://leetcode-cn.com/problems/unique-paths-ii/solution/jian-dan-dpbi-xu-miao-dong-by-sweetiee/