LeetCode Hot100(65/100)——64. 最小路径和

文章目录

题目链接

LeetCode 64. 最小路径和

题目说明

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

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

示例 1:

复制代码
输入: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

约束条件:

  • m == grid.length
  • n == grid[i].length
  • 1 <= m, n <= 200
  • 0 <= grid[i][j] <= 200

问题分析

这是一道经典的动态规划问题。我们需要从起点 (0,0) 走到终点 (m-1, n-1),每次只能向右或向下移动,目标是找到路径和最小的路径。
问题分析
路径限制
目标函数
解法思路
只能向右移动
只能向下移动
最小化路径和
动态规划
空间优化

解法一:二维动态规划

算法原理

使用二维数组 dp[i][j] 表示从起点 (0,0) 到达位置 (i,j) 的最小路径和。

状态转移方程:

  • 当 i = 0, j = 0 时:dp[0][0] = grid[0][0]
  • 当 i = 0 时(第一行):dp[0][j] = dp[0][j-1] + grid[0][j]
  • 当 j = 0 时(第一列):dp[i][0] = dp[i-1][0] + grid[i][0]
  • 其他情况:dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + grid[i][j]

思路流程图

i,j
完成
开始
初始化dp数组
dp_0_0 = grid_0_0
填充第一行
填充第一列
遍历剩余单元格
dp_i_j = min

dp_i-1_j, dp_i_j-1

  • grid_i_j
    返回dp_m-1_n-1
    结束

Java代码实现

java 复制代码
class Solution {
    public int minPathSum(int[][] grid) {
        int m = grid.length;
        int n = grid[0].length;
        
        // 创建dp数组
        int[][] dp = new int[m][n];
        
        // 初始化起点
        dp[0][0] = grid[0][0];
        
        // 初始化第一行
        for (int j = 1; j < n; j++) {
            dp[0][j] = dp[0][j - 1] + grid[0][j];
        }
        
        // 初始化第一列
        for (int i = 1; i < m; i++) {
            dp[i][0] = dp[i - 1][0] + grid[i][0];
        }
        
        // 填充其余位置
        for (int i = 1; i < m; i++) {
            for (int j = 1; j < n; j++) {
                dp[i][j] = Math.min(dp[i - 1][j], dp[i][j - 1]) + grid[i][j];
            }
        }
        
        return dp[m - 1][n - 1];
    }
}

复杂度分析

  • 时间复杂度:O(m × n),需要遍历整个网格一次
  • 空间复杂度:O(m × n),需要额外的二维数组存储dp值

解法二:一维动态规划(空间优化)

算法原理

观察状态转移方程,我们发现计算 dp[i][j] 时只需要 dp[i-1][j](上方)和 dp[i][j-1](左方)的值。因此可以使用一维数组进行空间优化。

使用一维数组 dp[j] 表示当前行到达第 j 列的最小路径和。在更新过程中:

  • dp[j] 更新前存储的是上一行第 j 列的值(相当于 dp[i-1][j]
  • dp[j-1] 存储的是当前行第 j-1 列的值(相当于 dp[i][j-1]

优化思路图

二维数组 m×n
观察依赖关系
只依赖上方和左方
使用一维数组
空间复杂度 O_n_

Java代码实现

java 复制代码
class Solution {
    public int minPathSum(int[][] grid) {
        int m = grid.length;
        int n = grid[0].length;
        
        // 只需要一维数组
        int[] dp = new int[n];
        
        // 初始化第一行
        dp[0] = grid[0][0];
        for (int j = 1; j < n; j++) {
            dp[j] = dp[j - 1] + grid[0][j];
        }
        
        // 逐行更新
        for (int i = 1; i < m; i++) {
            // 更新当前行的第一列
            dp[0] = dp[0] + grid[i][0];
            
            // 更新当前行的其他列
            for (int j = 1; j < n; j++) {
                dp[j] = Math.min(dp[j], dp[j - 1]) + grid[i][j];
            }
        }
        
        return dp[n - 1];
    }
}

复杂度分析

  • 时间复杂度:O(m × n),仍需遍历整个网格
  • 空间复杂度:O(n),只需要长度为 n 的一维数组

解法三:原地修改(终极优化)

算法原理

如果允许修改原数组,可以直接在 grid 数组上进行动态规划,无需额外空间。

Java代码实现

java 复制代码
class Solution {
    public int minPathSum(int[][] grid) {
        int m = grid.length;
        int n = grid[0].length;
        
        // 初始化第一行
        for (int j = 1; j < n; j++) {
            grid[0][j] += grid[0][j - 1];
        }
        
        // 初始化第一列
        for (int i = 1; i < m; i++) {
            grid[i][0] += grid[i - 1][0];
        }
        
        // 填充其余位置
        for (int i = 1; i < m; i++) {
            for (int j = 1; j < n; j++) {
                grid[i][j] += Math.min(grid[i - 1][j], grid[i][j - 1]);
            }
        }
        
        return grid[m - 1][n - 1];
    }
}

复杂度分析

  • 时间复杂度:O(m × n)
  • 空间复杂度:O(1),不需要额外空间

解法对比

解法对比
二维DP
时间: O_m×n_
空间: O_m×n_
易理解
一维DP
时间: O_m×n_
空间: O_n_
空间优化
原地修改
时间: O_m×n_
空间: O_1_
修改原数组

示例演示

以示例 1 为例:grid = [[1,3,1],[1,5,1],[4,2,1]]

二维DP过程

复制代码
初始网格:          dp数组构建过程:
1  3  1           1  4  5
1  5  1    →      2  7  6
4  2  1           6  8  7

最小路径和 = 7
路径: 1→3→1→1→1

状态转移示例

(0,0)

值:1

路径和:1
(0,1)

值:3

路径和:4
(1,0)

值:1

路径和:2
(1,1)

值:5

路径和:7
(2,2)

值:1

路径和:7

总结

本题是动态规划的经典问题,核心在于找到状态转移方程。三种解法的时间复杂度相同,但空间复杂度逐步优化:

  1. 二维DP:最直观,适合初学者理解
  2. 一维DP:空间优化,实际应用推荐
  3. 原地修改:极致优化,但会修改原数组

在实际面试中,建议先说明二维DP思路,再提出空间优化方案,展现对算法的深入理解。

相关推荐
CoderCodingNo2 小时前
【NOIP】2011真题解析 luogu-P1003 铺地毯 | GESP三、四级以上可练习
算法
iFlyCai3 小时前
C语言中的指针
c语言·数据结构·算法
查古穆3 小时前
栈-有效的括号
java·数据结构·算法
再一次等风来3 小时前
近场声全息(NAH)仿真实现:从阵列实值信号到波数域重建
算法·matlab·信号处理·近场声全息·nah
汀、人工智能3 小时前
16 - 高级特性
数据结构·算法·数据库架构·图论·16 - 高级特性
你撅嘴真丑3 小时前
[蓝桥杯 2025 省 B] 生产车间 与 装修报价
职场和发展·蓝桥杯
大熊背3 小时前
利用ISP离线模式进行分块LSC校正的方法
人工智能·算法·机器学习
XWalnut3 小时前
LeetCode刷题 day4
算法·leetcode·职场和发展
蒸汽求职4 小时前
机器人软件工程(Robotics SDE):特斯拉Optimus落地引发的嵌入式C++与感知算法人才抢夺战
大数据·c++·算法·职场和发展·机器人·求职招聘·ai-native
AI成长日志4 小时前
【笔面试算法学习专栏】双指针专题·简单难度两题精讲:167.两数之和II、283.移动零
学习·算法·面试