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

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

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

示例 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

memoziation

  1. class Solution:
  2. def uniquePaths(self, m: int, n: int) -> int:
  3. def f(m, n, memo = {}):
  4. key = (m, n)
  5. if key in memo:
  6. return memo[key]
  7. if m == 0 or n == 0:
  8. return 0
  9. if m == 1 and n == 1:
  10. return 1
  11. memo[key] = f(m - 1, n, memo) + f(m, n - 1, memo)
  12. return memo[key]
  13. return f(m, n)

tabulation

  1. class Solution:
  2. def uniquePaths(self, m: int, n: int) -> int:
  3. dp = [[0] * n for _ in range(m)]
  4. for i in range(m):
  5. for j in range(n):
  6. if i == 0 or j == 0:
  7. dp[m][n] = 1
  8. else:
  9. dp[m][n] = dp[m - 1][n] + dp[m][n - 1]
  10. return dp[-1]
  1. class Solution {
  2. public int uniquePaths(int m, int n) {
  3. int[][] f = new int[m][n];
  4. for (int i = 0; i < m; ++i) {
  5. f[i][0] = 1;
  6. }
  7. for (int j = 0; j < n; ++j) {
  8. f[0][j] = 1;
  9. }
  10. for (int i = 1; i < m; ++i) {
  11. for (int j = 1; j < n; ++j) {
  12. f[i][j] = f[i - 1][j] + f[i][j - 1];
  13. }
  14. }
  15. return f[m - 1][n - 1];
  16. }
  17. }

解法二:组合数

从m+n-2中选出m-1步向右,剩余的向下。

  1. class Solution {
  2. public int uniquePaths(int m, int n) {
  3. long ans = 1;
  4. for (int x = n, y = 1; y < m; ++x, ++y) {
  5. ans = ans * x / y;
  6. }
  7. return (int) ans;
  8. }
  9. }

相关题目

63. 不同路径 II

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

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

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

62. 不同路径 - 图2

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

示例 1:

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

示例 2:
62. 不同路径 - 图4

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

提示:

m == obstacleGrid.length
n == obstacleGrid[i].length
1 <= m, n <= 100
obstacleGrid[i][j] 为 0 或 1

  1. class Solution:
  2. def uniquePathsWithObstacles(self, obstacleGrid: List[List[int]]) -> int:
  3. def f(m, n, memo={}):
  4. key = (m, n)
  5. if key in memo:
  6. return memo[key]
  7. if obstacleGrid[m - 1][n - 1]:
  8. return 0
  9. if m == 0 or n == 0:
  10. return 0
  11. if m == 1 and n == 1:
  12. return 1
  13. memo[key] = f(m - 1, n, memo) + f(m, n - 1, memo)
  14. return memo[key]
  15. m, n = len(obstacleGrid), len(obstacleGrid[0])
  16. return f(m, n)

64. 最小路径和

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

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

示例 1:
62. 不同路径 - 图5

输入:grid = [[1,3,1],[1,5,1],[4,2,1]]
输出:7
解释:因为路径 1→3→1→1→1 的总和最小。
示例 2:

输入:grid = [[1,2,3],[4,5,6]]
输出:12

  1. class Solution:
  2. def minPathSum(self, grid: List[List[int]]) -> int:
  3. if not grid or not grid[0]:
  4. return 0
  5. m, n = len(grid), len(grid[0])
  6. dp = [[0] * n for _ in range(m)]
  7. dp[0][0] = grid[0][0]
  8. for i in range(1, m):
  9. dp[i][0] = dp[i - 1][0] + grid[i][0]
  10. for j in range(1, n):
  11. dp[0][j] = dp[0][j - 1] + grid[0][j]
  12. for i in range(1, m):
  13. for j in range(1, n):
  14. dp[i][j] = min(dp[i - 1][j], dp[i][j - 1]) + grid[i][j]
  15. return dp[m - 1][n - 1]

DP空间复杂度可以压缩至O(n),即只存储上一行的n个dp值。

  1. class Solution:
  2. def minPathSum(self, grid):
  3. dp = [float('inf')] * (len(grid[0])+1)
  4. dp[1] = 0
  5. for row in grid:
  6. for idx, num in enumerate(row):
  7. dp[idx + 1] = min(dp[idx], dp[idx + 1]) + num
  8. return dp[-1]