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

相关推荐
We་ct14 分钟前
LeetCode 300. 最长递增子序列:两种解法从入门到优化
开发语言·前端·javascript·算法·leetcode·typescript
VelinX18 分钟前
【个人学习||agent底层】01创建基础的发送和模型建立联系
学习
wayz1134 分钟前
Day 9 :随机森林调参与时间序列交叉验证
算法·随机森林·机器学习
️是7839 分钟前
信息奥赛一本通—编程启蒙(3371:【例64.2】 生日相同)
开发语言·c++·算法
ZPC821039 分钟前
ROS2 快过UDP的方法
python·算法·机器人
知识分享小能手41 分钟前
ECharts入门学习教程,从入门到精通,ECharts高级功能(6)
前端·学习·echarts
周末也要写八哥1 小时前
最长递增子序列典型应用题目详解
数据结构·算法
GISer_Jing1 小时前
Jinger独自勇闯Microsoft AI TourShanghai
学习·新浪微博
chudonghao1 小时前
[UE学习笔记][基于源码] 控制器、Pawn、相机的控制关系
笔记·学习·ue5
Fabarta技术团队1 小时前
务实、灵活——枫清科技财务单证智能审核方案 以AI自学习驱动审核提效与规则进化
人工智能·科技·学习