穿越障碍:最小路径和的高效算法比较【python力扣题64】

2024-04-25 12:36

本文主要是介绍穿越障碍:最小路径和的高效算法比较【python力扣题64】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

作者介绍:10年大厂数据\经营分析经验,现任大厂数据部门负责人。
会一些的技术:数据分析、算法、SQL、大数据相关、python
欢迎加入社区:码上找工作
作者专栏每日更新:
LeetCode解锁1000题: 打怪升级之旅
python数据分析可视化:企业实战案例
备注说明:方便大家阅读,统一使用python,带必要注释,公众号 数据分析螺丝钉 一起打怪升级

题目描述

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

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

输入格式
  • 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

方法一:动态规划

解题步骤
  1. 定义状态:创建一个同样大小的二维数组 dp,其中 dp[i][j] 表示到达点 (i, j) 的最小路径和。
  2. 初始化状态:第一行和第一列的元素只能由它的左边或上边来,所以是累加当前行或列的值。
  3. 状态转移:对于其他位置,dp[i][j] 由它的左边和上边的较小值加上当前网格值得到,即 dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + grid[i][j]
  4. 返回结果dp[m-1][n-1] 即为最小路径和。
完整的规范代码
def minPathSum(grid):"""使用动态规划解决最小路径和问题:param grid: List[List[int]], 网格:return: int, 最小路径和"""m, n = len(grid), len(grid[0])dp = [[0]*n for _ in range(m)]dp[0][0] = grid[0][0]for i in range(1, m):dp[i][0] = dp[i-1][0] + grid[i][0]for j in range(1, n):dp[0][j] = dp[0][j-1] + grid[0][j]for i in range(1, m):for j in range(1, n):dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + grid[i][j]return dp[m-1][n-1]# 示例调用
print(minPathSum([[1,3,1],[1,5,1],[4,2,1]
]))  # 输出: 7
print(minPathSum([[1,2,3],[4,5,6]
]))  # 输出: 12
算法分析
  • 时间复杂度:(O(m * n)),需要遍历整个网格一次。
  • 空间复杂度:(O(m * n)),使用了一个同样大小的二维数组。

方法二:空间优化的动态规划

解题步骤
  1. 使用一维数组:只用一个长度为 n 的数组来保存当前行的 dp 值。
  2. 迭代更新:每次更新时,dp[j] 更新为 dp[j](从上一行继承下来的,即上方)和 dp[j-1](当前行左边的,即左方)中的较小值加上当前点的值。
完整的规范代码
def minPathSum(grid):"""使用一维数组进行动态规划,空间优化版本:param grid: List[List[int]], 网格:return: int, 最小路径和"""m, n = len(grid), len(grid[0])dp = [0] * ndp[0] = grid[0][0]for j in range(1, n):dp[j] = dp[j-1] + grid[0][j]for i in range(1, m):dp[0] += grid[i][0]for j in range(1, n):dp[j] = min(dp[j-1], dp[j]) + grid[i][j]return dp[n-1]# 示例调用
print(minPathSum([[1,3,1],[1,5,1],[4,2,1]
]))  # 输出: 7
print(minPathSum([[1,2,3],[4,5,6]
]))  # 输出: 12
算法分析
  • 时间复杂度:(O(m * n)),需要遍历整个网格一次。
  • 空间复杂度:(O(n)),使用了一个长度为列数 n 的数组。

方法三:递归 + 记忆化

解题步骤
  1. 递归定义:定义一个递归函数,用于计算到达 (i, j) 的最小路径和。
  2. 记忆化存储:使用一个字典或数组来存储已经计算过的结果,避免重复计算。
完整的规范代码
def minPathSum(grid):"""使用递归和记忆化搜索解决最小路径和问题:param grid: List[List[int]], 网格:return: int, 最小路径和"""from functools import lru_cachem, n = len(grid), len(grid[0])@lru_cache(None)def dfs(i, j):if i == 0 and j == 0:return grid[i][j]if i < 0 or j < 0:return float('inf')return grid[i][j] + min(dfs(i-1, j), dfs(i, j-1))return dfs(m-1, n-1)# 示例调用
print(minPathSum([[1,3,1],[1,5,1],[4,2,1]
]))  # 输出: 7
print(minPathSum([[1,2,3],[4,5,6]
]))  # 输出: 12
算法分析
  • 时间复杂度:(O(m * n)),每个点最多计算一次,利用记忆化避免重复计算。
  • 空间复杂度:(O(m * n)),记忆化需要的空间。

方法四:从终点到起点的动态规划

解题步骤
  1. 反向动态规划:从网格的右下角开始,向左上角逐步计算。
  2. 更新规则:每个点的最小路径和取决于其右边和下边的点的最小路径和。
完整的规范代码
def minPathSum(grid):"""使用反向动态规划解决最小路径和问题:param grid: List[List[int]], 网格:return: int, 最小路径和"""m, n = len(grid), len(grid[0])for i in range(m-2, -1, -1):grid[i][n-1] += grid[i+1][n-1]for j in range(n-2, -1, -1):grid[m-1][j] += grid[m-1][j+1]for i in range(m-2, -1, -1):for j in range(n-2, -1, -1):grid[i][j] += min(grid[i+1][j], grid[i][j+1])return grid[0][0]# 示例调用
print(minPathSum([[1,3,1],[1,5,1],[4,2,1]
]))  # 输出: 7
print(minPathSum([[1,2,3],[4,5,6]
]))  # 输出: 12
算法分析
  • 时间复杂度:(O(m * n)),需要遍历整个网格一次。
  • 空间复杂度:(O(1)),直接在输入网格上进行修改,不需要额外空间。

方法五:改进的BFS

解题步骤
  1. 队列实现BFS:使用队列来实现广度优先搜索,每次处理一层。
  2. 累计最小和:使用额外的二维数组来保存到每个点的最小路径和。
  3. 优先队列优化:使用优先队列(小顶堆)来优先处理当前路径和最小的节点,以快速找到最小路径和。
完整的规范代码
from heapq import heappush, heappopdef minPathSum(grid):"""使用改进的BFS和优先队列解决最小路径和问题:param grid: List[List[int]], 网格:return: int, 最小路径和"""m, n = len(grid), len(grid[0])minHeap = [(grid[0][0], 0, 0)]  # (cost, x, y)costs = [[float('inf')] * n for _ in range(m)]costs[0][0] = grid[0][0]while minHeap:cost, x, y = heappop(minHeap)for dx, dy in [(1, 0), (0, 1)]:nx, ny = x + dx, y + dyif 0 <= nx < m and 0 <= ny < n:new_cost = cost + grid[nx][ny]if new_cost < costs[nx][ny]:costs[nx][ny] = new_costheappush(minHeap, (new_cost, nx, ny))return costs[m-1][n-1]# 示例调用
print(minPathSum([[1,3,1],[1,5,1],[4,2,1]
]))  # 输出: 7
print(minPathSum([[1,2,3],[4,5,6]
]))  # 输出: 12
算法分析
  • 时间复杂度:(O(m * n \log(m * n))),每个节点可能多次进入堆。
  • 空间复杂度:(O(m * n)),用于存储路径成本和堆结构。

不同算法的优劣势对比

特征方法一: 动态规划方法二: 空间优化DP方法三: 递归+记忆化方法四: 反向DP方法五: BFS+优先队列
时间复杂度(O(m * n))(O(m * n))(O(m * n))(O(m * n))(O(m * n \log(m * n)))
空间复杂度(O(m * n))(O(n))(O(m * n))(O(1))(O(m * n))
优势直观,易理解空间效率高避免重复计算,减少计算次数不需要额外空间,原地修改可以更快地找到最小路径和
劣势空间占用高仅限于列优化需要辅助空间存储递归状态修改输入数据计算和空间复杂度较高

应用示例

机器人导航系统
在自动化仓库或智能制造系统中,机器人需要找到成本最低的路径来移动货物或执行任务。动态规划方法可以有效地计算出从起点到终点的最低成本路径,提高系统的效率和响应速度。此外,实时路径规划系统可以利用优先队列优化的BFS来快速调整路径,以应对动态变化的环境条件,如临时障碍或优先级任务。

这篇关于穿越障碍:最小路径和的高效算法比较【python力扣题64】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/934695

相关文章

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python正则表达式匹配和替换的操作指南

《Python正则表达式匹配和替换的操作指南》正则表达式是处理文本的强大工具,Python通过re模块提供了完整的正则表达式功能,本文将通过代码示例详细介绍Python中的正则匹配和替换操作,需要的朋... 目录基础语法导入re模块基本元字符常用匹配方法1. re.match() - 从字符串开头匹配2.

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

通过Docker容器部署Python环境的全流程

《通过Docker容器部署Python环境的全流程》在现代化开发流程中,Docker因其轻量化、环境隔离和跨平台一致性的特性,已成为部署Python应用的标准工具,本文将详细演示如何通过Docker容... 目录引言一、docker与python的协同优势二、核心步骤详解三、进阶配置技巧四、生产环境最佳实践

Python一次性将指定版本所有包上传PyPI镜像解决方案

《Python一次性将指定版本所有包上传PyPI镜像解决方案》本文主要介绍了一个安全、完整、可离线部署的解决方案,用于一次性准备指定Python版本的所有包,然后导出到内网环境,感兴趣的小伙伴可以跟随... 目录为什么需要这个方案完整解决方案1. 项目目录结构2. 创建智能下载脚本3. 创建包清单生成脚本4

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

python获取指定名字的程序的文件路径的两种方法

《python获取指定名字的程序的文件路径的两种方法》本文主要介绍了python获取指定名字的程序的文件路径的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 最近在做项目,需要用到给定一个程序名字就可以自动获取到这个程序在Windows系统下的绝对路径,以下

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Python实现批量CSV转Excel的高性能处理方案

《Python实现批量CSV转Excel的高性能处理方案》在日常办公中,我们经常需要将CSV格式的数据转换为Excel文件,本文将介绍一个基于Python的高性能解决方案,感兴趣的小伙伴可以跟随小编一... 目录一、场景需求二、技术方案三、核心代码四、批量处理方案五、性能优化六、使用示例完整代码七、小结一、

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e