【每日天学习一点算法 2026/03/31】不同路径

每日天学习一点算法 2026/03/31

题目:不同路径

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

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

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

动态规划

这个题有点像爬楼梯

我们假设 dp[i][j] 是到达位置 (i, j) 的方法总数,那么在未触及边界的情况下 dp[i][j] = dp[i - 1][j] + dp[i][j - 1]

typescript 复制代码
function uniquePaths(m: number, n: number): number {
  const dp = new Array(m).fill(new Array(n).fill(1)) // 准备 dp 数组
  for (let i = 0; i < m; i++) {
    for (let j = 0; j < n; j++) {
      if (i === 0 && j === 0) continue // 第一个格子直接跳过
      if (i === 0) {
        // 没有上一行 
        dp[i][j] = dp[i][j - 1]
        continue
      }
      if (j === 0) {
        // 没有上一列
        dp[i][j] = dp[i - 1][j]
        continue
      }
      dp[i][j] = dp[i - 1][j] + dp[i][j - 1]
    }
  }
  return dp[m - 1][n - 1]
};

我们可以注意到,每次只会用到 dp[i][j - 1]dp[i - 1][j]

dp[i][j - 1]好办,因为我们是一行一行的遍历的,所以我们只用记录上一次遍历的dp 值就行了

dp[i - 1][j]就比较麻烦了,我们需要一个数组来记录上一行的 dp

typescript 复制代码
function uniquePaths(m: number, n: number): number {
  let topDps = []
  let leftDp = 1
  for (let i = 0; i < m; i++) {
    const arr = []
    for (let j = 0; j < n; j++) {
      if (i === 0 && j === 0) {
        // 第一行的第一个元素直接填入 1 
        arr.push(1)
        continue
      }
      if (i === 0) {
        // 第一行全都只有一种走法
        arr.push(leftDp)
        continue
      }
      if (j === 0) {
        // 第一列走法都一样
        leftDp = topDps[j]
        arr.push(leftDp)
        continue
      }
      // 记录其他位置走法
      leftDp = leftDp + topDps[j]
      arr.push(leftDp)
    }
    // 每行遍历完,重新赋值 leftDp 和 topDps
    leftDp = topDps[0]
    topDps = arr
  }
  return topDps[n - 1]
};
组合数学

我们可以知道,走到左下角一共要往下走 m - 1 ,往右走 n - 1 步,这个问题就可以转换成总共要走 m + n - 2 步,其中 m - 1 步的走法不一样,一共有多少种走法也就是:

由于没有 JS 没有阶乘函数,我们可以写一个阶乘方法,也可以写一个计算组合数的方法,我们这里直接写计算组合数方法,我们直接看C(m, n)更好理解一点:

我们把分子的乘法修改一些就可以变成 (m - n + 1)(m - n + 2)...(m - n + (n - 1))(m - n + n)

刚好对应分子的阶乘 1 * 2 * ... * (n - 1) * n

这样直接循环就可以得出组合数结果

typescript 复制代码
function uniquePaths(m: number, n: number): number {
  // 组合数计算方法
  function compose (m: number, n: number) {
    n = Math.min(n, m - n) // 取较小的一边减少循环
    let res = 1
    // 循环计算结果
    for (let i = 1; i <= n; i++) {
      res = res * (m - n + i) / i
    }
    return res
  }
    
  // 返回 C(m + n - 2, m - 1) 的组合数
  return compose(m + n - 2, m - 1)
};

题目来源:力扣(LeetCode)

相关推荐
Darkwanderor2 小时前
搜索优化——迭代加深dfs
c++·算法·深度优先·迭代加深
计算机安禾2 小时前
【数据结构与算法】第17篇:串(String)的高级模式匹配:KMP算法
c语言·数据结构·学习·算法·visual studio code·visual studio·myeclipse
大萌神Nagato2 小时前
力扣HOT100 Q146LRU缓存
算法·leetcode·缓存
源码之家2 小时前
大数据毕业设计汽车推荐系统 Django框架 可视化 协同过滤算法 数据分析 大数据 机器学习(建议收藏)✅
大数据·python·算法·django·汽车·课程设计·美食
nianniannnn2 小时前
力扣 3.无重复字符的最长子串
c++·算法·leetcode
IT大师兄吖3 小时前
flux-2-Klein-BFS-换头换脸工作流 懒人整合包
算法·宽度优先
波哥学开发3 小时前
深入解析 BEV 图像色彩调整与伪彩色映射:从直方图统计到着色器实现
算法·图形学
西西弟3 小时前
最短路径之Floyd算法(数据结构)
数据结构·算法
小O的算法实验室3 小时前
2026年SEVC,直觉模糊不确定环境下求解绿色多物品固定费用五维运输问题的多目标进化算法,深度解析+性能实测
算法·论文复现·智能算法·智能算法改进