1. leetcode 64. 最小路径和

给定一个包含非负整数的 m x n 网格,

请找出一条从左上角到右下角的路径,

使得路径上的数字总和为最小。

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

示例:

输入:

[

[1,3,1],

[1,5,1],

[4,2,1]

]

输出: 7

解释: 因为路径 1→3→1→1→1 的总和最小。

1.1. 暴力

cost(i,j)=grid[i][j]+min(cost(i+1,j),cost(i,j+1))

    def minPathSum_1(self, grid):
"""
暴力法
时间复杂度 :O(2^(m+n})。每次移动最多可以有两种选择。
空间复杂度 :O(m+n)。递归的深度是 m+n。
:param grid:
:return:
"""
len_i, len_j = len(grid), len(grid[0]) def _helper(grid, i, j, len_i, len_j):
if i == len_i or j == len_j:
return float('inf')
if i == len_i - 1 and j == len_j - 1:
return grid[i][j]
return grid[i][j] + min(_helper(grid, i+1, j, len_i, len_j),
_helper(grid, i, j+1, len_i, len_j),) return _helper(grid, 0, 0, len_i, len_j)

1.2. 二维动态规划

新建一个额外的dp数组,与原矩阵大小相同。在这个矩阵中,dp(i, j)表示从坐标 (i, j)到右下角的最小路径权值。

初始化右下角的dp值为对应的原矩阵值,然后去填整个矩阵,对于每个元素考虑移动方式,获得最小路径;

如下递推公式:

dp(i,j)=grid(i,j)+min(dp(i+1,j),dp(i,j+1))

注意边界情况。

    def minPathSum_2(self, grid):
"""
二维动态规划
时间复杂度 :O(mn)。遍历整个矩阵恰好一次。
空间复杂度 :O(n)。额外的矩阵。
:param grid:
:return:
"""
len_i, len_j = len(grid), len(grid[0])
dp = [[0 for _ in range(len_j)] for _ in range(len_i)] for i in range(len_i - 1, -1, -1):
for j in range(len_j - 1, -1, -1):
if i == len_i - 1 and j != len_j - 1:
dp[i][j] = grid[i][j] + dp[i][j+1]
elif j == len_j -1 and i != len_i - 1:
dp[i][j] = grid[i][j] + dp[i+1][j]
elif j != len_j -1 and i != len_i - 1:
dp[i][j] = grid[i][j] + min(dp[i+1][j], dp[i][j+1])
else:
dp[i][j] = grid[i][j]
return dp[0][0]

1.2.1. 动态规划优化

在上一解法中使用数组记录了所有的路径值,但实际只需要记录一行的数据,所以可以做如下优化:

   def minPathSum_3(self, grid):
"""
动态规划 一维数组
上面的动态规划保存了整个矩阵,
但实际只用保存一行/列的数据即可
时间复杂度 :O(mn)。遍历整个矩阵恰好一次。
空间复杂度 :O(n)。额外的一维数组,和一行大小相同。
:param grid:
:return:
"""
len_i, len_j = len(grid), len(grid[0])
dp = [0]*len_j for i in range(len_i - 1, -1, -1):
for j in range(len_j - 1, -1, -1):
if i == len_i - 1 and j != len_j - 1:
dp[j] = grid[i][j] + dp[j+1]
elif j == len_j - 1 and i != len_i - 1:
dp[j] = grid[i][j] + dp[j]
elif i != len_i - 1 and j != len_j - 1:
dp[j] = grid[i][j] + min(dp[j], dp[j+1])
else:
dp[j] = grid[i][j]
return dp[0]

1.2.2. 优化2

当然也可以使用数组本身的空间,使得所需的额外空间为零。

需要注意的是改变数组本身的行为不要放在其它解决方案测试之前;

否则使用copy.deepcopy()。

    def minPathSum_4(self, grid: [[int]]) -> int:
"""
动态规划
不使用额外空间
时间复杂度:O(MN)
空间复杂度:O(1)
:param grid:
:return:
"""
for i in range(len(grid)):
for j in range(len(grid[0])):
if i == j == 0: continue
elif i == 0: grid[i][j] = grid[i][j - 1] + grid[i][j]
elif j == 0: grid[i][j] = grid[i - 1][j] + grid[i][j]
else: grid[i][j] = min(grid[i - 1][j], grid[i][j - 1]) + grid[i][j]
return grid[-1][-1]

2. 完整代码及执行结果


# coding:utf-8

__author__ = "sn"

"""
64. 最小路径和
给定一个包含非负整数的 m x n 网格,
请找出一条从左上角到右下角的路径,
使得路径上的数字总和为最小。
说明:每次只能向下或者向右移动一步。 示例:
输入:
[
[1,3,1],
[1,5,1],
[4,2,1]
]
输出: 7
解释: 因为路径 1→3→1→1→1 的总和最小。
""" """
思路:
暴力穷举
动态规划 """ class Solution:
def minPathSum_1(self, grid):
"""
暴力法
时间复杂度 :O(2^(m+n})。每次移动最多可以有两种选择。
空间复杂度 :O(m+n)。递归的深度是 m+n。
:param grid:
:return:
"""
len_i, len_j = len(grid), len(grid[0]) def _helper(grid, i, j, len_i, len_j):
if i == len_i or j == len_j:
return float('inf')
if i == len_i - 1 and j == len_j - 1:
return grid[i][j]
return grid[i][j] + min(_helper(grid, i+1, j, len_i, len_j),
_helper(grid, i, j+1, len_i, len_j),) return _helper(grid, 0, 0, len_i, len_j) def minPathSum_2(self, grid):
"""
二维动态规划
时间复杂度 :O(mn)。遍历整个矩阵恰好一次。
空间复杂度 :O(n)。额外的矩阵。
:param grid:
:return:
"""
len_i, len_j = len(grid), len(grid[0])
dp = [ [0] * len_j ] * len_i for i in range(len_i - 1, -1, -1):
for j in range(len_j - 1, -1, -1):
if i == len_i - 1 and j != len_j - 1:
dp[i][j] = grid[i][j] + dp[i][j+1]
elif j == len_j -1 and i != len_i - 1:
dp[i][j] = grid[i][j] + dp[i+1][j]
elif j != len_j -1 and i != len_i - 1:
dp[i][j] = grid[i][j] + min(dp[i+1][j], dp[i][j+1])
else:
dp[i][j] = grid[i][j]
return dp[0][0] def minPathSum_3(self, grid):
"""
动态规划 一维数组
上面的动态规划保存了整个矩阵,
但实际只用保存一行/列的数据即可
时间复杂度 :O(mn)。遍历整个矩阵恰好一次。
空间复杂度 :O(n)。额外的一维数组,和一行大小相同。
:param grid:
:return:
"""
len_i, len_j = len(grid), len(grid[0])
dp = [0]*len_j for i in range(len_i - 1, -1, -1):
for j in range(len_j - 1, -1, -1):
if i == len_i - 1 and j != len_j - 1:
dp[j] = grid[i][j] + dp[j+1]
elif j == len_j - 1 and i != len_i - 1:
dp[j] = grid[i][j] + dp[j]
elif i != len_i - 1 and j != len_j - 1:
dp[j] = grid[i][j] + min(dp[j], dp[j+1])
else:
dp[j] = grid[i][j]
return dp[0] def minPathSum_4(self, grid: [[int]]) -> int:
"""
动态规划
不使用额外空间
时间复杂度:O(MN)
空间复杂度:O(1)
:param grid:
:return:
"""
for i in range(len(grid)):
for j in range(len(grid[0])):
if i == j == 0: continue
elif i == 0: grid[i][j] = grid[i][j - 1] + grid[i][j]
elif j == 0: grid[i][j] = grid[i - 1][j] + grid[i][j]
else: grid[i][j] = min(grid[i - 1][j], grid[i][j - 1]) + grid[i][j]
return grid[-1][-1] import timeit
from collections import Iterable def processing_func(cls, func_name, *ar, **kw):
func = getattr(cls, func_name) if isinstance(ar[0], Iterable):
res = func(*ar[0])
else:
res = func()
# 打印执行结果
print('执行结果:', res) import timeit
def test_func(cls, *ar): # 方法执行耗时
time_cost = dict() # 获取并执行Solution类中的解决方法
func_list = [x for x in dir(cls) if not x.startswith('__')]
print('\r\n', "共计有个方法:"%(len(func_list)), func_list)
# 设置参数 # 依次执行Solution类中的方法
for i, _ in enumerate(func_list, 1):
# 跳过
#if i == 1: continue
# 设置参数
func = getattr(cls, _)
# print(cls, func_num, func, ar) # 打印方法说明文档
print("\r\n", "*" * 40, "\r\n方法[%s]:%s\r\n说明:%s" % (i, _, func.__doc__.strip()),) # 执行方法并记录执行时长
t = timeit.timeit(stmt="processing_func(so, '{}', para)".format(_),
setup='from __main__ import processing_func, so, para',
number= 1
)
time_cost[_] = t
print('\r\n执行时长:', *time_cost.items(), sep='\r\n') if __name__ == "__main__":
# 实例化解决方案类
so = Solution() # 参数设定
li = [
[1, 3, 1],
[1, 5, 1],
[4, 2, 1],
]
para = (li,) test_func(so, para)
pass

2.1. 执行结果

 共计有个方法: ['minPathSum_1', 'minPathSum_2', 'minPathSum_3', 'minPathSum_4']

 ****************************************
方法[1]:minPathSum_1
说明:暴力法
时间复杂度 :O(2^(m+n})。每次移动最多可以有两种选择。
空间复杂度 :O(m+n)。递归的深度是 m+n。
:param grid:
:return:
执行结果: 7 ****************************************
方法[2]:minPathSum_2
说明:二维动态规划
时间复杂度 :O(mn)。遍历整个矩阵恰好一次。
空间复杂度 :O(n)。额外的矩阵。
:param grid:
:return:
执行结果: 7 ****************************************
方法[3]:minPathSum_3
说明:动态规划 一维数组
上面的动态规划保存了整个矩阵,
但实际只用保存一行/列的数据即可
时间复杂度 :O(mn)。遍历整个矩阵恰好一次。
空间复杂度 :O(n)。额外的一维数组,和一行大小相同。
:param grid:
:return:
执行结果: 7 ****************************************
方法[4]:minPathSum_4
说明:动态规划
不使用额外空间
时间复杂度:O(MN)
空间复杂度:O(1)
:param grid:
:return:
执行结果: 7 执行时长:
('minPathSum_1', 0.00013762672739037798)
('minPathSum_2', 3.638943978457454e-05)
('minPathSum_3', 2.332656396447088e-05)
('minPathSum_4', 2.239350140589196e-05)

leetcode 64. 最小路径和 动态规划系列的更多相关文章

  1. [LeetCode] 64. 最小路径和 ☆☆☆(动态规划)

    描述 给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小. 说明:每次只能向下或者向右移动一步. 示例: 输入:[  [1,3,1], [1,5,1 ...

  2. LeetCode 64. 最小路径和(Minimum Path Sum) 20

    64. 最小路径和 64. Minimum Path Sum 题目描述 给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小. 说明: 每次只能向下或 ...

  3. Java实现 LeetCode 64 最小路径和

    64. 最小路径和 给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小. 说明:每次只能向下或者向右移动一步. 示例: 输入: [ [1,3,1], ...

  4. LeetCode 64最小路径和

    题目 给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小. 说明:每次只能向下或者向右移动一步. 示例: 输入: [   [1,3,1], [1,5 ...

  5. Leetcode——64. 最小路径和

    题目描述:题目链接 同样对于这个问题,我们可以考虑用动态规划来解决. 解决动态规划常见的三个步骤: 1:问题的归纳.对于 i,j 位置上的最短路径可以用d[ i ][ j ]表示. 2:归纳递推式:d ...

  6. leetcode 64. 最小路径和Minimum Path Sum

    很典型的动态规划题目 C++解法一:空间复杂度n2 class Solution { public: int minPathSum(vector<vector<int>>&am ...

  7. [LeetCode]64. 最小路径和(DP)

    题目 给定一个无序的整数数组,找到其中最长上升子序列的长度. 示例: 输入: [10,9,2,5,3,7,101,18] 输出: 4 解释: 最长的上升子序列是 [2,3,7,101],它的长度是 4 ...

  8. Leetcode之动态规划(DP)专题-64. 最小路径和(Minimum Path Sum)

    Leetcode之动态规划(DP)专题-64. 最小路径和(Minimum Path Sum) 给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小. ...

  9. Leetcode题目64.最小路径和(动态规划-中等)

    题目描述: 给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小. 说明:每次只能向下或者向右移动一步. 示例: 输入: [ [1,3,1], [1, ...

随机推荐

  1. 使用SFTP连接Centos

    1.centos已经配置好了SFTP,直接使用root用户连接就可以,模式选SFTP即可. 2.虽然端口号没有填写,默认端口号是22 3.可能还是会遇到无法访问的问题,可以进行iptables防火墙的 ...

  2. babel配置文件.babelrc

    Babel是一个广泛应用的转码器,可以将ES6代码转为ES5代码,从而在现有环境执行.意味着可以使用ES6编写程序,而不用担心现有环境是否支持. Babel的配置文件是.babelrc,存放在项目的根 ...

  3. gulp常用插件之gulp-rev-css-url使用

    更多gulp常用插件使用请访问:gulp常用插件汇总 gulp-rev-css-url这是一款用于在gulp-rev之后覆盖js.css文件中的URL进行替换. 更多使用文档请点击访问gulp-rev ...

  4. Linux connect: Network is unreachable

    在虚拟机中ping,发现网络不通: [root@node01 ~]# ping 114.114.114.114 connect: Network is unreachable 发生此问题时,环境如下: ...

  5. [ERR] Node goodsleep.vip:6379 is not empty. Either the node already knows other nodes (check with CLUSTER NODES) or contains some key in database 0.

    解决方案 以前的cluster节点信息 保留 要删除 dump.rdb node.conf集群启动时自动生成文件

  6. vue-routerV3.1版本报错:message: "Navigating to current location ("/home") is not allowed",

    出现这个错误的原因是,在路由跳转的时候两次push的path地址相同 解决方法两种: 1.切换版本回3.0版本 2.在你引了vue-router的js文件里加上如下代码即可 import VueRou ...

  7. vue之项目打包部署到服务器

    这是今年的第一篇博客.整理一下vue如何从项目打包到部署服务器,给大家做下分享,希望能给大家带来或多或少的帮助,喜欢的大佬们可以给个小赞,如果有问题也可以一起讨论下. 第一步:这是很关键的一步.打开项 ...

  8. css实现聊天气泡效果

      --------------------------------------- css功能强大,能实现很多炫 酷的效果,今天给大家分享 用css3绘制聊天气泡的方法: -------------- ...

  9. yii2 生成随机字符串

    uuid uuid use Faker\Provider\Uuid; Uuid::uuid(); yii自带 生成32位字符串 Yii::$app->getSecurity()->gene ...

  10. CSS的列表样式和网页背景

    CSS的列表样式 1. 设置title和列表 HTML: <!DOCTYPE html><html lang="en"><head>    &l ...