动态规划解决2xN网格最大路径和问题

动态规划解决2xN网格最大路径和问题

本文深入探讨了如何在2xn的网格中,从a[0]到b[-1]寻找最大路径和的动态规划方法。文章详细阐述了dp状态定义、基线条件及状态转移方程,并通过python代码示例展示了从初始实现到优化后的完整过程。重点强调了代码结构优化技巧,旨在提升实现效率和可读性,同时保持算法的o(n)时间复杂度。

2xN网格最大路径和问题详解

问题描述

假设我们有两个长度为N的一维整数数组A和B,它们可以被视为一个2xN的网格。A代表第一行,B代表第二行。我们需要从网格的左上角元素A[0]出发,移动到右下角元素B[N-1],且每次只能向右移动一格或向下移动一格。目标是找到一条路径,使得路径上所有元素的和最大。

例如,对于N=3:网格结构如下:A[0] A[1] A[2]B[0] B[1] B[2]

可能的路径示例:A[0] -> A[1] -> A[2] -> B[2] (非法,不能从A[2]直接到B[2],必须经过B[1]或从A[2]向下到B[2])A[0] -> A[1] -> B[1] -> B[2]A[0] -> B[0] -> B[1] -> B[2]A[0] -> A[1] -> B[1] (非法,终点是B[N-1])

正确的路径示例:A[0] -> A[1] -> A[2] (然后必须向下) -> B[2]A[0] -> A[1] (然后向下) -> B[1] -> B[2]A[0] (然后向下) -> B[0] -> B[1] -> B[2]

动态规划方法

这个问题可以通过动态规划(Dynamic Programming, DP)有效地解决。我们可以定义一个二维DP表 dp,其中 dp[row][col] 表示到达网格中 (row, col) 位置时的最大路径和。

由于我们只有两行,DP表可以定义为 dp[2][N]。

dp[0][i] 表示到达数组A中 A[i] 位置时的最大路径和。dp[1][i] 表示到达数组B中 B[i] 位置时的最大路径和。

基线条件

起始点 A[0]:到达 A[0] 的最大路径和就是 A[0] 本身。dp[0][0] = A[0]

起始点 B[0]:要到达 B[0],只能从 A[0] 向下移动。dp[1][0] = dp[0][0] + B[0]

状态转移方程

对于 i > 0 的情况:

计算 dp[0][i] (到达 A[i]):要到达 A[i],只能从 A[i-1] 向右移动。dp[0][i] = dp[0][i-1] + A[i]

计算 dp[1][i] (到达 B[i]):要到达 B[i],有两种可能的路径:

从 B[i-1] 向右移动:dp[1][i-1] + B[i]从 A[i] 向下移动:dp[0][i] + B[i]我们选择这两种路径中和最大的一个。dp[1][i] = max(dp[1][i-1] + B[i], dp[0][i] + B[i])

最终结果就是 dp[1][N-1],即到达 B[N-1] 的最大路径和。

初始实现与优化

以下是根据上述逻辑编写的Python实现,并对其进行优化。

初始实现示例

def max_path_sum_initial(A, B):    N = len(A)    # dp[0][i] 存储到达 A[i] 的最大和    # dp[1][i] 存储到达 B[i] 的最大和    dp = [[0 for _ in range(N)] for _ in range(2)]    # 基线条件    dp[0][0] = A[0]    # 计算第一行 A 的路径和    for i in range(1, N):        dp[0][i] = dp[0][i - 1] + A[i]    # 计算 B[0] 的路径和 (注意这里在原始代码中被放在了循环内,是不必要的重复计算)    # dp[1][0] = dp[0][0] + B[0]    # 计算第二行 B 的路径和    # 原始代码中的 B[0] 计算在这里    for i in range(N): # 循环从0开始以包含 B[0] 的计算        if i == 0:            dp[1][0] = dp[0][0] + B[0] # 第一次迭代计算 B[0]        else:            dp[1][i] = max(dp[1][i - 1] + B[i], dp[0][i] + B[i])    return dp[1][N - 1]# 示例测试# A = [1, 2, 3]# B = [4, 5, 6]# print(max_path_sum_initial(A, B)) # 期望输出: 1+2+3+6 = 12 或 1+4+5+6 = 16# 1 -> 2 -> 3 -> (down) -> 6 = 12# 1 -> 2 -> (down) -> 5 -> 6 = 14# 1 -> (down) -> 4 -> 5 -> 6 = 16

上述 max_path_sum_initial 函数是基于原始问题的代码逻辑重构的,它展示了原始代码中可能存在的计算结构上的冗余。

优化建议

原始实现虽然在算法逻辑上是正确的,但在代码结构上存在两处可以优化的点,这些优化主要提升了代码的简洁性和效率,但不会改变算法的渐近时间复杂度。

提前计算 dp[1][0]:dp[1][0] 的值 dp[0][0] + B[0] 是一个常量,它不依赖于任何循环迭代。因此,应该在主循环开始之前计算一次,而不是在循环内部(即使是条件判断)重复计算。

合并循环:dp[0][i] 和 dp[1][i] 的计算可以在同一个循环中完成。dp[1][i] 的计算依赖于 dp[0][i] 和 dp[1][i-1]。由于 dp[0][i] 是在当前迭代中计算的,并且 dp[1][i-1] 已经在前一个迭代中计算完毕,所以将它们放在一个循环中是完全可行的。这可以减少代码量并提高可读性。

优化后的实现

def max_path_sum_optimized(A, B):    N = len(A)    if N == 0:        return 0 # 处理空数组情况    dp = [[0 for _ in range(N)] for _ in range(2)]    # 基线条件:计算 dp[0][0] 和 dp[1][0]    dp[0][0] = A[0]    dp[1][0] = dp[0][0] + B[0] # 优化点1:提前计算 dp[1][0]    # 合并循环,计算 dp[0][i] 和 dp[1][i]    for i in range(1, N): # 优化点2:合并循环        dp[0][i] = dp[0][i - 1] + A[i]        dp[1][i] = max(dp[1][i - 1] + B[i], dp[0][i] + B[i])    return dp[1][N - 1]# 示例测试A_test = [1, 2, 3]B_test = [4, 5, 6]print(f"优化后函数计算结果: {max_path_sum_optimized(A_test, B_test)}") # 期望输出: 16A_test2 = [10, -5, 20]B_test2 = [1, 10, 5]print(f"优化后函数计算结果2: {max_path_sum_optimized(A_test2, B_test2)}")# Path 1: A[0]->A[1]->A[2]->B[2] = 10 + (-5) + 20 + 5 = 30# Path 2: A[0]->A[1]->B[1]->B[2] = 10 + (-5) + 10 + 5 = 20# Path 3: A[0]->B[0]->B[1]->B[2] = 10 + 1 + 10 + 5 = 26# Max is 30

复杂度分析

时间复杂度: 算法遍历了N次,每次迭代执行常数次操作。因此,时间复杂度为 O(N)空间复杂度: 我们使用了2xN的DP表来存储中间结果。因此,空间复杂度为 O(N)

空间复杂度优化(进阶)

注意到在计算 dp[0][i] 和 dp[1][i] 时,我们只依赖于 dp[0][i-1] 和 dp[1][i-1] 以及当前的 A[i] 和 B[i]。这意味着我们实际上不需要存储整个2xN的DP表。我们可以只用常数空间来存储前一个状态的值。

def max_path_sum_space_optimized(A, B):    N = len(A)    if N == 0:        return 0    # 使用两个变量存储前一个位置的最大和    # current_a_sum 对应 dp[0][i-1]    # current_b_sum 对应 dp[1][i-1]    # 初始化基线条件    prev_a_sum = A[0]    prev_b_sum = prev_a_sum + B[0]    for i in range(1, N):        # 计算当前 A[i] 的最大和        current_a_sum = prev_a_sum + A[i]        # 计算当前 B[i] 的最大和        # 它可以从 B[i-1] 过来,或者从 A[i] 向下过来        current_b_sum = max(prev_b_sum + B[i], current_a_sum + B[i])        # 更新 prev_a_sum 和 prev_b_sum 为当前值,为下一次迭代做准备        prev_a_sum = current_a_sum        prev_b_sum = current_b_sum    return prev_b_sum# 示例测试A_test = [1, 2, 3]B_test = [4, 5, 6]print(f"空间优化后函数计算结果: {max_path_sum_space_optimized(A_test, B_test)}") # 期望输出: 16

通过空间优化,我们将空间复杂度降低到了 O(1),因为我们只使用了几个变量来存储状态,而不是整个DP表。

总结

本教程详细介绍了使用动态规划解决2xN网格中最大路径和问题的方法。从问题定义到基线条件、状态转移方程,再到Python代码实现,我们逐步构建并优化了解决方案。通过将 dp[1][0] 的计算移到循环外部以及合并两个独立的循环,我们提高了代码的简洁性和效率。最终,还展示了如何将空间复杂度从O(N)进一步优化到O(1),这在处理大规模输入时尤为重要。理解这些优化技巧对于编写高效且可维护的动态规划代码至关重要。

以上就是动态规划解决2xN网格最大路径和问题的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1382841.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月15日 00:35:23
下一篇 2025年12月15日 00:35:41

相关推荐

  • VSCode中Conda虚拟环境激活与使用疑难排解

    当在VSCode中遇到Conda虚拟环境无法正确激活,导致代码无法在指定环境中运行时,问题通常在于终端环境配置未能识别或加载正确的虚拟环境。本教程提供了一种直接通过导航至虚拟环境脚本目录并执行激活脚本的方法,以确保您的Python代码能够在预期的隔离环境中运行,解决终端提示符不显示环境名称的问题。 …

    好文分享 2025年12月15日
    000
  • python线程强制停止工作

    Python中无法强制终止线程,推荐使用标志位或Event事件实现协作式停止。例如通过设置布尔变量或threading.Event通知线程退出,避免资源泄漏;若需强制终止,可改用multiprocessing.Process及其terminate()方法。 Python 中线程一旦启动,不能直接强制…

    2025年12月15日
    000
  • 如何使用python中F-Strings字符串?

    F-Strings是Python 3.6+推荐的字符串格式化方法,通过f前缀和{}嵌入变量或表达式,支持表达式计算、数字日期格式化、转义及多行字符串,兼具简洁性、可读性与高效性。 F-Strings 是 Python 3.6+ 引入的一种字符串格式化方法,写法简洁、读起来直观,执行效率也高。它通过在…

    2025年12月15日
    000
  • python面向对象中多态的使用前提是什么?

    多态的前提是继承和方法重写,子类继承同一父类并重写其方法,通过父类引用调用同名方法时,根据实际对象执行不同逻辑,如Dog和Cat继承Animal并重写speak(),make_sound函数接受Animal类型参数,传入不同子类对象输出“汪汪”或“喵喵”,体现“同一种调用,不同行为”,Python的…

    2025年12月15日
    000
  • python中字典与json相互转换的方法

    字典与JSON字符串可通过json模块相互转换:使用json.dumps()将字典转为JSON字符串,支持indent和ensure_ascii等参数美化输出;json.loads()将合法JSON字符串解析为字典;文件操作则用json.dump()写入、json.load()读取;注意键必须为字符…

    2025年12月15日
    000
  • python中reduce函数和map函数的区别有哪些?

    map用于逐元素转换,返回等长序列;reduce用于累积聚合,返回单一值。前者是内置函数,后者需导入functools模块。 reduce 和 map 都是 Python 中用于处理可迭代对象的函数,但它们的作用和使用方式有本质区别。下面从功能、返回值、使用场景等方面说明它们的不同。 功能上的区别 …

    2025年12月15日
    000
  • python使用Plotly实现动画设计

    答案:使用Plotly制作动画需组织好按时间划分的数据帧,通过go.Figure的frames参数定义每帧图形,配合sliders和play按钮实现播放控制,并设置统一坐标轴范围与过渡效果以提升流畅性。 用Python结合Plotly制作动画,关键在于理解其帧(frames)和更新逻辑。Plotly…

    2025年12月15日
    000
  • ​python的id函数如何判断分片产生的列表?

    分片操作会创建新列表对象,其id与原列表不同,表明两者为独立对象,修改互不影响,但无法通过id判断是否由分片产生。 Python 中的 id() 函数返回对象的唯一标识符(通常是内存地址)。当你对列表进行分片操作时,会创建一个全新的列表对象,因此它的 id 值与原列表不同。 分片产生新对象 列表分片…

    2025年12月15日
    000
  • 在python中调用staticmethod用到参数

    静态方法不依赖实例或类,通过@staticmethod定义,可接收任意参数用于工具函数、计算等,如MathUtils.add(3, 5)返回8,Validator.is_adult(20)返回True,TemperatureConverter转换温度,适用于无需访问属性的逻辑。 在 Python 中…

    2025年12月15日
    000
  • python中不同推导式怎么写

    Python推导式提供简洁语法创建序列或映射,主要包括列表、字典、集合推导式及生成器表达式。列表推导式通过[表达式 for 变量 in 可迭代对象 if 条件]生成列表,如[x2 for x in range(10)]创建0到9的平方列表;添加条件可筛选结果,如[x2 for x in range(…

    2025年12月15日
    000
  • 如何使用Python timeit模块?

    timeit模块用于测量小段代码执行时间,通过多次运行取最小耗时以减少误差。使用timeit.timeit()函数,传入代码字符串和运行次数number(默认100万次)即可测试性能差异。 Python的timeit模块用来测量小段代码的执行时间,特别适合对比不同实现方式的性能差异。它通过多次运行代…

    2025年12月15日
    000
  • 类继承如何在python面向对象中实现?有什么好处?

    Python中通过类名后加父类实现继承,子类可重写或扩展父类方法,支持多层与多重继承,提升代码复用、可维护性与扩展性,并实现多态。 在 Python 面向对象编程中,类继承通过子类继承父类的属性和方法来实现代码复用和结构化设计。你只需要在定义类时,在类名后面加上父类的名字即可完成继承。 如何实现类继…

    2025年12月15日
    000
  • Python multiprocessing.Pool进程状态诊断与超时排查

    本文旨在解决python `multiprocessing.pool`在执行异步任务时可能出现的超时问题,特别是当`pool.get()`抛出`timeouterror`时,难以确定具体是哪个子进程导致阻塞。我们将深入探讨`multiprocessing.process`对象的`exitcode`属…

    2025年12月15日
    000
  • Python模块条件导入:优化复杂项目结构中的依赖管理

    本教程旨在解决python项目中因不同程序入口导致共享模块导入路径失败的`modulenotfounderror`问题。核心策略是将按需加载的模块导入语句封装到函数内部,实现“惰性导入”。这确保了依赖只在被明确调用时加载,有效避免了不必要的导入错误,同时保持了代码的清晰性和项目结构的合理性,无需借助…

    2025年12月15日
    000
  • Python高效生成与存储大规模内存访问轨迹的实践指南

    本文旨在解决在python中为内存模拟器生成和存储大规模内存访问轨迹时遇到的性能与内存瓶颈。通过深入分析`print()`函数和内存存储的局限性,文章提出并详细阐述了直接利用文件写入流的高效策略。教程将提供示例代码,指导读者如何以指定格式(如`0x12345678 w`)高效地将数据写入文件,从而优…

    2025年12月15日
    000
  • PyArrow Decimal128 精度管理:避免数据损失的舍入策略

    本文深入探讨了在pandas与pyarrow `decimal128`类型操作中遇到的精度管理挑战。当执行涉及`decimal128`类型的计算时,pyarrow会自动扩展精度,导致直接类型转换可能引发数据损失异常。文章详细解释了这一机制,并提供了一种通过在类型转换前进行显式舍入来有效解决数据损失问…

    2025年12月15日
    000
  • 如何在Python中动态创建全局变量

    本文将深入探讨如何在Python中根据变量的值动态创建全局变量。我们将介绍使用内置的`globals()`函数这一推荐方法,它允许开发者直接操作当前模块的全局符号表,从而实现灵活的变量命名和赋值。文章还将对比并解释为何应避免使用`exec()`等方法,并提供清晰的示例代码和最佳实践建议,以确保代码的…

    2025年12月15日
    000
  • Python中利用数据模型对象实现运算符重载与Pyright类型检查兼容性指南

    本文探讨了在python中通过数据模型对象实现灵活且避免重复代码的运算符重载策略。针对每个运算符具有相同多重重载的场景,我们设计了`apply`和`op`两个辅助类。然而,这种模式在pyright类型检查器中对中缀运算符存在兼容性问题。教程将详细介绍问题根源,并提供通过在`op`类中添加`__cal…

    2025年12月15日
    000
  • python中callable的对象有哪些?

    可调用对象是能使用()操作符执行的对象,包括函数、类、定义了__call__方法的实例、方法及内置函数等,通过callable(obj)可判断其是否可调用。 在 Python 中,callable 指的是可以被调用的对象,也就是能使用 () 操作符执行的对象。可以通过内置函数 callable(ob…

    2025年12月15日
    000
  • Python元组语法深度解析:何时需要括号以及列表推导式中的特殊考量

    本文深入探讨python元组的语法规则,重点解析何时需要使用括号以及其在表达式和列表推导式中的关键作用。我们将通过实例阐明元组的隐式创建(打包)、显式定义,以及括号如何影响运算符优先级和解决代码歧义,从而帮助开发者更好地理解和运用python元组。 1. Python元组的基本构成与隐式创建 在Py…

    2025年12月15日
    000

发表回复

登录后才能评论
关注微信