首页 > 其他分享 >以Top-Down思维去解决问题——递归

以Top-Down思维去解决问题——递归

时间:2024-08-29 16:36:49浏览次数:10  
标签:问题 函数 递归 Top alist Down 阶乘 自上而下

目录


递归和for循环(迭代法)很像,都是通过循环去完成一件事。

采用Top-Down思维去设计的递归结构,又会比for多一些不同的能力。多什么能力?

递归的基础

先复习一下递归,递归的定义:递归(英语:Recursion),又译为递回,在数学与计算机科学中,是指在函数的定义中使用函数自身的方法。

image

递归的本质就在:递去归来 两个流程中。 初学者刚接触会有点抽象,下面通过一些案例来认识。

假设需要你实现一个阶乘的计算函数,
阶乘的定义:
5的阶乘是 5*4*3*2*1=120

def factorial(number):
    if number == 1:
        return 1
    else:
        return number * factorial(number - 1)

print(factorial(5))
// 120

递归需要考虑三个条件:

  1. 将问题拆分成一个重复使用的子问题;
  2. 注意,这个子问题和问题的规模无关;
  3. 必须包含终止条件 (终止条件即初始状态)。

递归的底层实现(不是重点)

递归的底层实现不是本文的重点,了解一下就行。

递归在编程语言的底层实现通常依赖于调用栈(call stack):

  • 调用栈

    • 每次函数调用时,程序会将函数的参数、局部变量和返回地址等信息压入栈帧。
    • 当递归函数调用自身时,会创建新的栈帧压入栈中。
    • 函数执行完毕后,栈帧被弹出,返回控制权给调用者。
  • 基线条件

    • 递归必须有终止条件,否则会导致栈溢出(stack overflow)。
    • 每次递归调用都应该向基线条件靠近。
  • 内存管理

    • 调用栈通常在内存的“栈区”分配。栈的大小有限制,过多的递归调用可能导致栈溢出。
    • 有些语言提供机制来增加栈的大小,但一般不推荐依赖深层递归。

总结:

  • 递归实现的效率和安全性与具体语言的特性和编译器的优化密切相关。

  • 递归的底层实现,就是把相关变量数据(缓存)处理后,一层一层的压入栈,等到了基准条件后,在逐层拿出处理。

计算机眼里的递归:
计算机使用栈来记录正在调用的函数,叫调用栈

image

有个局部变量 number 记录当前值。

递归的应用场景

  • 处理任意多层事情的场景,都可以考虑用递归。

  • 当问题和子问题具有递推关系,比如杨辉三角、计算阶乘。

  • 具有递归性质的数据结构,比如链表、树、图。

  • 反向性问题,比如取反操作。

编程中 两种解决问题的思维

这个才是本文重点要学习的。

当面对未来未知的情况时,考虑使用使用自上而下解决问题的思维。

两种编写计算函数的方法:

  • 自下而上(Bottom - Up)
    类似循环
  • 自上而下 (Top - Down)
    递归思想

两者区别?
在计算函数时,自下而上和自上而下是两种不同的思维方式和实现策略:

在计算函数时,特别是像阶乘这样的递归函数,可以使用两种主要的实现方式来实现递归计算:自下而上(bottom-up)和自上而下(top-down)。这些方法各有优缺点,理解它们有助于选择适合的实现方式来解决特定的问题。以下是对这两种实现方式的解释:

自下而上(Bottom-Up)

自下而上方法,也称为 迭代方法动态规划方法,是指从最小的子问题开始逐步构建解决方案,直到解决原始问题。这种方法通常用于动态规划算法中,但也可以用于一些简单的递归问题。

实现思路:

  1. 定义最小问题: 从问题的最小子问题开始解决。例如,在计算阶乘时,可以从 0!1! 开始。
  2. 逐步构建: 使用迭代或循环逐步计算更大的问题的解。
  3. 更新和存储: 将每个子问题的解存储起来,以便后续使用。

还是以计算阶乘的案例去介绍,自下而上实现方式:

def factorial_bottom_up(n):
    if n < 0:
        raise ValueError("Factorial is not defined for negative numbers")
    result = 1
    for i in range(2, n + 1):
        result *= i
    return result

# 示例用法
print(factorial_bottom_up(5))  # 输出 120

解释:

  • 从 1 开始迭代计算 2 到 n 的阶乘。
  • 通过逐步乘以每个数字来更新结果,最终得到 n!

自上而下(Top-Down)

自上而下方法也称为 递归方法,是指从解决问题的最上层开始,递归地解决较小的子问题。这种方法在处理递归问题时非常自然(但可能存在重复计算的子问题,有些可以优化)。

实现思路:

  1. 定义主问题: 从问题的最上层开始解决。
  2. 递归分解: 将主问题递归地分解为较小的子问题。
  3. 基本情况: 定义递归的基本情况,以停止递归。

还是以计算阶乘的案例,展示自上而下实现:

def factorial_top_down(n):
    if n < 0:
        raise ValueError("Factorial is not defined for negative numbers")
    if n == 0 or n == 1:
        return 1
    return n * factorial_top_down(n - 1)

# 示例用法
print(factorial_top_down(5))  # 输出 120

解释:

  • n 开始,递归调用 factorial_top_down(n - 1),直到达到基本情况。
  • 在基本情况时返回 1,逐层返回乘积结果。

总结

  • 自下而上:通常是迭代的方法,逐步构建解决方案,适用于动态规划和需要避免重复计算的情况。它通常较为高效,尤其是在解决子问题重复计算时。

  • 自上而下:通常是递归的方法,直观地解决问题,但可能会有较高的时间复杂度和空间复杂度,尤其在处理大规模问题时。(可以通过记忆化(Memoization)来优化性能)

自上而下的思考过程——求和案例

一般来说,自下而上的实现过程比较好理解,所以这里多列举一些自上而下的案例帮助思考,

自上而下的编程思考过程:

  1. 把你正在写的函数想象成是别人实现过的函数。
  2. 辨别子问题。
  3. 看看你在子问题上调用函数时会发生什么,然后以此为基础继续。

求和案例
假设我们要写一个 sum 函数,计算数组中所有数的和。如果给函数传入数组[1,2,3,4,5],那么它会返回这些数的和15。

我们需要做的第一件事就是想象已经有人实现了 sum 函数。(当然,你可能会有点难以接受,毕竟写函数是我们自己,怎么能假设别人写好了呢! 但可以试着忘掉这一点,先假装 sum 函数已经实现好了。)

接下来,来辨别子问题。比起科学,这个过程更像是艺术,只要你多练习就能进步。 在这个例子中,可以认为子问题是数组[2,3,4,5],即原数组中除第一个数以外的元素。

最后,来看看在子问题上调用 sum 函数会发生什么 ?
如果 sum 函数“已被正确实现”并且子问题是 [2,3,4,5],那么调用 sum([2,3,4,5])时会发生什么呢?会得到2+3+4+5的和,也就是14。

而要求[1,2,3,4,5]的和,只需向 sum([2,3,4,5])的结果再加上1即可。

请使用编程语言实现一下:

mylist = [1, 2, 3, 4, 5]
def mysum(alist):

    if len(alist) == 1:
        return alist[0]
    else:
		# alist[-1] = alist[len(alist)-1]
        alist[0] += alist[-1]

    return mysum(alist[0:len(alist) - 1])

print(mysum(mylist))
# 15

台阶问题 案例

为什么需要用递归?
image

image

请写出代码:

todo

易位构词生成 案例

这个是一个很实用的案例,之前想将多个pyload 以不同位置组合成一个整体,就遇到这个难题。

image

请写出代码:

todo

标签:问题,函数,递归,Top,alist,Down,阶乘,自上而下
From: https://www.cnblogs.com/mysticbinary/p/18371511

相关文章

  • 1-1 Markdown学习
    AI为我提供的Markdown相关问题解答如下:Markdown的详细语法:Markdown是一种轻量级标记语言,它允许人们使用易读易写的纯文本格式编写文档,然后转换成有效的HTML。下面是一些基本的Markdown语法:标题:使用#表示标题,一级标题使用一个#,二级标题使用两个#,以此类推。例如,#一级标题,##二......
  • Luogu P4425 转盘 题解 [ 黑 ] [ 线段树 ] [ 贪心 ] [ 递归 ]
    转盘:蒟蒻的第一道黑,这题是贪心和线段树递归合并的综合题。贪心破环成链的trick自然不用多说。首先观察题目,很容易发现一个性质:只走一圈的方案一定最优。这个很容易证,因为再绕一圈回来标记前面的和等前面的标记完之后继续走是等价的,并且再绕一圈甚至可能更劣。于是,我们只用走......
  • 【Markdown笔记】设置字体颜色——转载https://blog.csdn.net/u012028275/article/det
     【Markdown笔记】设置字体颜色dadalaohua于2021-04-0517:53:19发布阅读量5.7w 收藏 293点赞数103分类专栏: Markdown笔记 文章标签: markdown latex html版权GitCode开源社区文章已被社区收录加入社区Markdown笔记专......
  • gyp GET https://nodejs.org/download/release/v20.15.0/node-v20.15.0-headers.tar.g
    如图我执行yarn关于node会报错:gyphttpGEThttps://nodejs.org/download/release/v20.15.0/node-v20.15.0-headers.tar.gzgyphttpfetchGEThttps://nodejs.org/download/release/v20.15.0/node-v20.15.0-headers.tar.gzattempt1failedwithETIMEDOUTgypWARNins......
  • Markdown学习
    一.ai返回的内容1.深入浅出的讲解Markdown及其详细语法Markdown是一种轻量级的标记语言,它允许人们使用易读易写的纯文本格式编写文档,然后转换成有效的XHTML(或者HTML)。Markdown的语法简洁明了,使得文档编写更加高效和专注。Markdown的基本语法包括:标题:使用#符号表示标题,#的数量......
  • Markdown语法
    Markdown学习一.标题#➕空格➕标题最多六级三级标题四级标题五级标题六级标题二.字体Hello,World!**Hello,World!*Hello,World!***Hello,World!~~三.引用>➕空格世界上只有一种英雄主义,那就是认清了生活的真相后还依然热爱它。四.分割线---或***五.......
  • Markdown学习
    1.哪些内容是你掌握的?哪些内容是你没有掌握的?使用AI推荐的工具或者你喜欢的工具实践一下没有掌握的内容。在AI推荐的推荐Markdown工具中,Typora和Ulysses是我比较掌握的工具,而Github我不太掌握。1.从Github上找到并下载一个项目2.以记事本打开内部附带的.md文件3.以支持Markd......
  • day13: 第六章 二叉树part01 |二叉树的前序遍历,后序遍历,中序遍历,(递归。层序(广度)跟
    二叉树递归三部曲:1.确定递归函数的参数和返回值。2.确定终止条件3.确定单层递归的逻辑144.二叉树的前序遍历:中左右,递归:classSolution{publicList<Integer>preorderTraversal(TreeNoderoot){List<Integer>res=newArrayList<Integer>();p......
  • 【北京迅为】龙芯iTOP-LS2K0500开发板快速启动手册-第3章 Windows安装串口终端
      LS2K0500采用龙芯2K0500处理器,基于龙芯自主指令系统(LoongArch)架构,片内集成64位LA264处理器核。实现ACPI、DVFS/DPM动态电源功耗管理等低功耗技术,支持多种电源级别和唤醒方式,可根据具体应用场景对芯片部分功能和高速接口进行动态时钟、电源开关控制,满足工控、网......
  • OpenCV(cv::getOptimalDFTSize())
    目录1.函数定义2.示例3.总结cv::getOptimalDFTSize()是OpenCV中的一个函数,用于返回最优的离散傅里叶变换(DFT)大小。具体来说,它帮助找到一个比给定大小更大的最优尺寸,用来加速傅里叶变换的计算。cv::getOptimalDFTSize()的功能是返回适合执行快速傅里叶变换(FFT)的最优......