【每日天学习一点算法 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)

相关推荐
_李小白4 小时前
【android opencv学习笔记】Day 28: 滤波算法之中值滤波器
android·opencv·学习
yuanyuan2o24 小时前
模型预训练:Hugging Face Transformers 基础
算法·ai·语言模型·自然语言处理·nlp·深度优先
杨充4 小时前
1.3 浮点型数据设计灵魂
开发语言·python·算法
妄想出头的工业炼药师5 小时前
GS slam mono
算法·开源
_日拱一卒6 小时前
LeetCode:207课程表
java·数据结构·算法·leetcode·职场和发展
飞翔中文网6 小时前
Java学习笔记之抽象类与接口(设计思想)
java·笔记·学习
土星碎冰机7 小时前
xxljob学习(大白话版本)
学习·运维开发
吃好睡好便好7 小时前
说说免疫力的维护
学习·生活
凉、介8 小时前
深入理解 ARMv8-A|处理器模式与寄存器
笔记·学习·嵌入式·arm