
本文深入探讨了经典的楼梯问题,即计算孩子以1、2或3步跳跃方式登上n级楼梯的所有可能方法数。文章详细介绍了两种动态规划解决方案:带有记忆化的递归方法和底向上迭代方法,并通过go语言示例代码演示了其实现细节、关键逻辑修正以及性能考量,旨在提供清晰、专业的教程指导。
假设一个孩子要跑上一个有 n 级台阶的楼梯,他每次可以跳1步、2步或3步。我们的目标是实现一个方法来计算孩子登上这 n 级台阶总共有多少种不同的方式。这是一个典型的动态规划问题,因为它具有重叠子问题和最优子结构特性。
动态规划的核心思想是避免重复计算。对于楼梯问题,要到达第 n 级台阶,孩子最后一步可能是从第 n-1 级跳1步,从第 n-2 级跳2步,或者从第 n-3 级跳3步。因此,到达第 n 级台阶的总方式数是到达这三级台阶的方式数之和。
我们可以定义 CountWays(n) 为到达第 n 级台阶的方式数,则有: CountWays(n) = CountWays(n-1) + CountWays(n-2) + CountWays(n-3)
基本情况(Base Cases):
为了避免重复计算,我们引入记忆化(Memoization),使用一个映射(map)或数组来存储已经计算过的结果。
Go语言实现及关键修正:
在Go语言中,map 在获取一个不存在的键时,会返回该值类型的零值。对于 int 类型,零值是 0。这一点在实现记忆化时尤为重要。
package main
import "fmt"
// CountWaysDP 使用递归和记忆化计算上楼梯的方式
// n: 目标台阶数
// memo: 用于存储已计算结果的映射
func CountWaysDP(n int, memo map[int]int) int {
// 基本情况:无法到达负数台阶
if n < 0 {
return 0
}
// 基本情况:到达第0级台阶(即起始位置),算作1种方式
if n == 0 {
return 1
}
// 检查是否已经计算过此结果
// 注意:Go的map在键不存在时返回零值(int为0)。
// 如果memo[n]为0,我们无法区分是未计算还是计算结果恰好为0。
// 但在此问题中,合法的方式数总是大于0。
// 因此,如果memo[n] > 0,则表示已经计算过且结果有效。
if memo[n] > 0 {
return memo[n]
}
// 递归计算并存储结果
memo[n] = CountWaysDP(n-1, memo) +
CountWaysDP(n-2, memo) +
CountWaysDP(n-3, memo)
return memo[n]
}
func main() {
memo := make(map[int]int) // 初始化记忆化映射
n := 10
result := CountWaysDP(n, memo)
fmt.Printf("到达 %d 级台阶共有 %d 种方式。\n", n, result)
// fmt.Println("记忆化映射内容:", memo) // 可以打印查看记忆化过程
}注意事项:
Remover
几秒钟去除图中不需要的元素
304
查看详情
迭代的动态规划方法通常从基本情况开始,逐步计算到目标问题。这种方法避免了递归调用的开销,通常在性能上更优。
Go语言实现:
package main
import "fmt"
// CountWaysIterative 使用迭代(底向上)方式计算上楼梯的方式
// n: 目标台阶数
func CountWaysIterative(n int) int {
if n < 0 {
return 0
}
if n == 0 {
return 1
}
// dp[i] 表示到达第 i 级台阶的方式数
// 需要 n+1 个元素,因为索引从0到n
dp := make([]int, n+1)
// 基本情况
dp[0] = 1 // 到达第0级台阶有1种方式(不跳)
// 从第1级台阶开始迭代计算
for i := 1; i <= n; i++ {
// 每次可以跳1、2或3步
// dp[i] = dp[i-1] + dp[i-2] + dp[i-3]
// 需要检查 i-k 是否越界
if i-1 >= 0 {
dp[i] += dp[i-1]
}
if i-2 >= 0 {
dp[i] += dp[i-2]
}
if i-3 >= 0 {
dp[i] += dp[i-3]
}
}
return dp[n]
}
func main() {
n := 10
result := CountWaysIterative(n)
fmt.Printf("到达 %d 级台阶共有 %d 种方式。\n", n, result)
// 另一种迭代实现方式(更紧凑)
// dp := make([]int, n+1)
// dp[0] = 1
// for i := 1; i <= n; i++ {
// if i >= 1 {
// dp[i] += dp[i-1]
// }
// if i >= 2 {
// dp[i] += dp[i-2]
// }
// if i >= 3 {
// dp[i] += dp[i-3]
// }
// }
// fmt.Println(dp[n])
}代码解释:
楼梯问题是动态规划的经典入门案例,它清晰地展示了如何通过分解问题为重叠子问题并存储中间结果来提高效率。
方法通常更高效,因为它避免了递归开销,且内存访问模式更为连续,尤其适合于键是连续整数的场景。对于此类问题,优先考虑使用切片(slice)而非映射(map)进行记忆化存储,以获得更好的性能。掌握这两种动态规划的实现方式,对于解决更复杂的组合优化问题至关重要。
以上就是动态规划解决楼梯问题:递归与迭代方法详解的详细内容,更多请关注其它相关文章!
相关文章:
J*a应用程序首次运行自动创建文件与目录的最佳实践
深入理解J*aScript Promise异步执行与微任务队列
yandex入口引擎手机版 yandex安卓版下载入口
Django AJAX 文件上传教程:解决图片无法保存到模型的常见问题
“在文档元素之后找到了标记”是什么错误? 检查并修复XML中多个根元素的3个方法
2306选座时如何选靠窗位置_12306选座靠窗座位查看方法解析
mysql如何分析事务日志_mysql事务日志分析方法
Win11如何开启讲述人功能 Win11屏幕阅读器(讲述人)开启与关闭【教程】
马斯克:Optimus 人形机器人复数形式为 Optimi
苹果手机如何防止被恶意App追踪
QQ邮箱官方网站登录入口_QQ邮箱网页版在线使用
怎么在html里运行vbs脚本_html中运行vbs脚本方法【教程】
ArrayList与LinkedList操作复杂度详解:遍历与修改
Kafka Streams中基于消息头条件过滤消息的实现指南
Python vgamepad库按键模拟:正确使用XUSB_BUTTON常量
win11 Snap Layouts怎么用 Win11窗口布局与分屏多任务高效指南【必学】
WooCommerce 购物车显示所有交叉销售商品教程
向日葵客户端怎么进行远程CentOS控制_向日葵客户端远程CentOS控制操作教程
铁路12306改签能改到更早的车次吗_铁路12306改签提前车次规则
如何使用Rector自动化升级旧代码_通过Composer安装和配置Rector进行代码重构
Yandex搜索引擎官方地址 俄罗斯网络世界的主要入口
三星ZFold5多任务卡顿_Samsung ZFold5流畅度提升
Win11怎么设置开机NumLock亮 Win11修改注册表InitialKeyboardIndicators值
J*a如何实现并发下载文件_J*a多线程IO性能优化案例
c++如何实现单例设计模式_c++线程安全的单例模式写法
Linux如何排查内存不足OOME问题_LinuxOOM分析教程
狙击外星人小游戏开始_狙击外星人小游戏立即开始
厨房不锈钢水槽发黑生锈怎么处理_水槽用可乐+锡纸2分钟抛亮如新
在WordPress中通过REST API访问受BasicAuth保护的站点内容
CSS自定义字体样式被系统字体替换怎么办_font-face方式指定font-display控制渲染策略
Win11怎么合并任务栏图标 Win11开启任务栏合并减少图标占空间【方法】
J*aScript中在Map循环中检测并处理空数组元素
Python:递归比较文件夹内容并找出特定类型文件的差异
解决PHP会话Cookie在跨域请求中不保留的问题
腾讯QQ邮箱官方网站_QQ邮箱网页版在线登录
PHP表单提交后函数重复执行的解决方案:管理$_POST数据
《北京人工智能产业白皮书(2025)》发布:全年核心产值预计突破 4500 亿元
TikTok网页版直接登录 TikTok网页端官方平台入口
C++编译期如何执行复杂计算_C++模板元编程(TMP)技巧与应用
Python Sounddevice 音频卡顿问题解析与队列数据安全处理
Mac怎么锁定备忘录_Mac备忘录加密设置教程
BetterDiscord插件中安全更新用户简介的实践指南
Composer的 "licenses" 命令如何帮助你遵守开源协议_检查项目依赖的许可证合规性
Go语言:非阻塞式判断标准输入(os.Stdin)是否有数据
如何在PHP中实现基于MySQL的动态分页查询
J*a中实现Go语言select通道多路复用机制
sublime如何优雅地处理行尾空格_sublime自动清理多余空白字符配置
Lar*el Form Request中唯一性验证在更新操作中的正确实现
品牌机怎么重装系统 联想/戴尔/惠普笔记本恢复出厂系统教程
单射、满射与双射的关系 一文理清所有逻辑