全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

动态规划解决楼梯问题:递归与迭代方法详解

本文深入探讨了经典的楼梯问题,即计算孩子以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):

  • 当 n
  • 当 n = 0 时,表示已经到达或无需移动(站在地面),这算作 1 种方式(即不跳)。

为了避免重复计算,我们引入记忆化(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) // 可以打印查看记忆化过程
}

注意事项:

  • 原始问题中 else if mm[n] > -1 的判断对于 map 在Go中是错误的,因为 map 对不存在的键返回 0,而 0 > -1 为真,会导致对未计算的 n 返回 0。正确的判断应该是 memo[n] > 0(因为方式数不可能为负或零,对于 n > 0 的台阶),或者更严谨地使用 if _, ok := memo[n]; ok 来检查键是否存在,并初始化 map 中的值为一个不可能的哨兵值(如 -1)来区分未计算和计算结果。但对于此问题,memo[n] > 0 足够。
  • 对于键是连续整数的动态规划问题,使用切片(slice)作为记忆化存储通常比 map 更高效,因为切片提供了O(1)的索引访问,且内存连续性更好。

动态规划方法二:迭代(底向上)

迭代的动态规划方法通常从基本情况开始,逐步计算到目标问题。这种方法避免了递归调用的开销,通常在性能上更优。

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])
}

代码解释:

  1. 创建一个切片 dp,大小为 n+1,其中 dp[i] 存储到达第 i 级台阶的方式数。
  2. 初始化 dp[0] = 1,表示到达第0级台阶有一种方式。
  3. 从 i = 1 循环到 n,计算 dp[i]。
  4. 对于每个 i,dp[i] 等于 dp[i-1] + dp[i-2] + dp[i-3]。在累加之前,需要确保 i-k 不会小于 0。例如,当 i=1 时,只能从 dp[0] 累加;当 i=2 时,可以从 dp[1] 和 dp[0] 累加。

总结

楼梯问题是动态规划的经典入门案例,它清晰地展示了如何通过分解问题为重叠子问题并存储中间结果来提高效率。

  • 递归与记忆化 方法直观地反映了问题的数学递推关系,但可能存在函数调用栈深度限制和一定的函数调用开销。
  • 迭代(底向上) 方法通常更高效,因为它避免了递归开销,且内存访问模式更为连续,尤其适合于键是连续整数的场景。对于此类问题,优先考虑使用切片(slice)而非映射(map)进行记忆化存储,以获得更好的性能。

掌握这两种动态规划的实现方式,对于解决更复杂的组合优化问题至关重要。


# go  # go语言  #   # ai  # if  # 递归  # int  # 循环  # 值类型 


相关文章: 网站制作大概要多少钱一个,做一个平台网站大概多少钱?  ,巨量百应是干嘛的?  网站建设制作、微信公众号,公明人民医院怎么在网上预约?  制作网站怎么制作,*游戏网站怎么搭建?  最好的网站制作公司,网购哪个网站口碑最好,推荐几个?谢谢?  公司网站的制作公司,企业网站制作基本流程有哪些?  建站之星在线版空间:自助建站+智能模板一键生成方案  猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?  实现虚拟支付需哪些建站技术支撑?  焦点电影公司作品,电影焦点结局是什么?  做企业网站制作流程,企业网站制作基本流程有哪些?  建站之星安装后如何自定义网站颜色与字体?  制作网站哪家好,cc、.co、.cm哪个域名更适合做网站?  宝塔建站助手安装配置与建站模板使用全流程解析  专业的网站制作设计是什么,如何制作一个企业网站,建设网站的基本步骤有哪些?  专业公司网站制作公司,用什么语言做企业网站比较好?  如何基于云服务器快速搭建个人网站?  北京企业网站设计制作公司,北京铁路集团官方网站?  网站海报制作教学视频教程,有什么免费的高清可商用图片网站,用于海报设计?  网站制作软件有哪些,制图软件有哪些?  网站制作外包价格怎么算,招聘网站上写的“外包”是什么意思?  网站网页制作电话怎么打,怎样安装和使用钉钉软件免费打电话?  湖南网站制作公司,湖南上善若水科技有限公司做什么的?  如何在橙子建站上传落地页?操作指南详解  小自动建站系统:AI智能生成+拖拽模板,多端适配一键搭建  装修招标网站设计制作流程,装修招标流程?  定制建站如何定义?其核心优势是什么?  香港网站服务器数量如何影响SEO优化效果?  深圳 网站制作,深圳招聘网站哪个比较好一点啊?  网站设计制作公司地址,网站建设比较好的公司都有哪些?  如何用狗爹虚拟主机快速搭建网站?  平台云上自助建站如何快速打造专业网站?  太原网站制作公司有哪些,网约车营运证查询官网?  如何在景安云服务器上绑定域名并配置虚拟主机?  如何通过山东自助建站平台快速注册域名?  简单实现Android文件上传  关于BootStrap modal 在IOS9中不能弹出的解决方法(IOS 9 bootstrap modal ios 9 noticework)  制作假网页,招聘网的薪资待遇,会有靠谱的吗?一面试又各种折扣?  如何高效配置IIS服务器搭建网站?  单页制作网站有哪些,朋友给我发了一个单页网站,我应该怎么修改才能把他变成自己的呢,请求高手指点迷津?  香港服务器网站生成指南:免费资源整合与高速稳定配置方案  如何快速搭建自助建站会员专属系统?  宝塔面板如何快速创建新站点?  电影网站制作价格表,那些提供免费电影的网站,他们是怎么盈利的?  如何在七牛云存储上搭建网站并设置自定义域名?  小型网站制作HTML,*游戏网站怎么搭建?  建站之星安装路径如何正确选择及配置?  购物网站制作公司有哪些,哪个购物网站比较好?  香港代理服务器配置指南:高匿IP选择、跨境加速与SEO优化技巧  黑客如何通过漏洞一步步攻陷网站服务器? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。