如何理解递归

news2024/11/26 0:52:00

在二叉树的题目中,我们难免会用到递归方法,递归思想很简单,但运用起来却因为抽象而难以理解。

理解递归的关键在于认识到它是一种解决问题的方法,允许函数直接或间接地调用自身。以下是对递归的概述以及如何理解它的几个要点:

1. 基本概念

递归是用一个函数调用其自身来解决问题。每次递归调用都会处理问题的一部分,直到达到一个基本情况(即停止条件)。

2. 结构

通常,递归包含两部分:

  • 基本情况(Base Case):这是停止递归的条件。没有这个条件,递归将无限进行,导致栈溢出。
  • 递归情况(Recursive Case):这是函数调用自身以解决更小的子问题。

3. 示例:阶乘

一个经典的递归示例是计算阶乘。定义阶乘的递归形式如下:

  • ( n! = n \times (n-1)! )(递归情况)
  • ( 0! = 1 )(基本情况)

其实现如下:

def factorial(n: int) -> int:
    if n == 0:  # 基本情况
        return 1
    else:  # 递归情况
        return n * factorial(n - 1)

在调用 factorial(5) 时,实际的调用过程是这样的:

  • factorial(5) 计算 5 * factorial(4)
  • factorial(4) 计算 4 * factorial(3)
  • factorial(3) 计算 3 * factorial(2)
  • factorial(2) 计算 2 * factorial(1)
  • factorial(1) 计算 1 * factorial(0)
  • factorial(0) 返回 1

4. 可视化递归

为了帮助理解递归,可以使用树结构来可视化。例如,当计算 factorial(5) 时,可以画出一棵树,显示每个函数调用如何分支到下一个调用。最终,每个分支都返回结果,汇总至最顶层的函数。

5. 递归的问题解决步骤

获取递归解法的一般步骤:

  1. 定义问题:了解要解决的具体问题。
  2. 找出基本情况:明确何时停止递归。
  3. 确定递归关系:如何将大问题拆分为更小的子问题。
  4. 通过示例理解执行过程:逐步追踪函数调用,以深入理解每一步的作用。

6. 递归 vs 迭代

  • 递归方法可以有更简洁和更具可读性的实现,但有时更容易导致性能问题(例如过多的函数调用可能导致栈溢出)。
  • 循环(或迭代)通常会更高效,尤其是在不需要存储调用栈的情况下。

7. 实践

解决各种问题(如遍历树、斐波那契数列、背包问题等)可以加强对递归的理解。实践是掌握递归最有效的方式。

我们来看看力扣144题目:二叉树的前序遍历
在这里插入图片描述
在这里插入图片描述
代码不好理解的话,可以在自己的电脑上运行下面的代码。

from typing import Optional, List  

# 定义二叉树节点类  
class TreeNode:  
    def __init__(self, val=0, left=None, right=None):  
        self.val = val  
        self.left = left  
        self.right = right  

# 定义Solution类并实现前序遍历方法  
class Solution:  
    def preorderTraversal(self, root: Optional[TreeNode]) -> List[int]:  
        # 打印当前节点的值  
        if not root:  
            print("当前节点: None")  
            return []  
        print(f"当前节点: {root.val}")  

        result = []  
        result.append(root.val)  # 添加根节点的值  
        print(f"当前结果: {result}")  # 打印结果  

        # 递归遍历左子树  
        left_result = self.preorderTraversal(root.left)  
        result.extend(left_result)  

        # 递归遍历右子树  
        right_result = self.preorderTraversal(root.right)  
        result.extend(right_result)  

        print(f"返回结果: {result}")  # 打印返回的结果  
        return result  

# 示例:创建一棵二叉树并运行前序遍历  
if __name__ == "__main__":  
    # 创建二叉树  
    #      1  
    #     / \
    #    2   3  
    #   / \
    #  4   5  
    root = TreeNode(1)  
    root.left = TreeNode(2)  
    root.right = TreeNode(3)  
    root.left.left = TreeNode(4)  
    root.left.right = TreeNode(5)  

    # 创建解决方案实例并调用前序遍历  
    solution = Solution()  
    result = solution.preorderTraversal(root)  

    # 输出最终结果  
    print(f"最终前序遍历结果: {result}")  # 输出应为 [1, 2, 4, 5, 3]

在这里插入图片描述
下面是图解递归算法:

前序遍历的顺序是:中左右。我们如何利用递归方法解决此道题目呢?我们可以假设一个简单的情况(root)不为空。
在这里插入图片描述
前序遍历,我们先把中值root.val添加到result中。接下来我们要处理左节点了,左节点也是要中左右,这个时候我们可以借助递归来处理这种重复的动作。
在这里插入图片描述

我们以下面的二叉树为例:
在这里插入图片描述
递归算法里其实就是在做三件事:
在这里插入图片描述
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2063398.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【算法】一文带你搞懂完全背包!(附背包问题总结)

理论基础 有N件物品和一个最多能背重量为W的背包。第i件物品的重量是weight[i],得到的价值是value[i] 。每件物品都有无限个(也就是可以放入背包多次),求解将哪些物品装入背包里物品价值总和最大。这就是完全背包问题。 完全背包…

【计算机组成原理】三、存储系统:1.存储器的分类、层次化结构、性能指标、基本组成(半导体、存储芯片基本原理)

三、存储系统(存储器层次结构) 文章目录 三、存储系统(存储器层次结构)1.存储器的分类1.1按在计算机中的作用(层次)❗多级存储结构(层次化结构)1.2按存储介质1.3按存取方式1.4按信息…

uniapp 小程序 设置按钮固定到页面的最下方

解决方案 我们在做小程序的时候,特别是页面是以列表的形式进行展示,并且页面必须还要新增数据时,这是就会在页面的底部加一个固定的新增按钮,点击新增按钮,弹出一个弹窗…然后进行下一步的业务逻辑操作,那…

Answer use of function tool by OpenAI assistant in Python

题意:“在 Python 中使用 OpenAI 助手的函数工具的用途” 问题背景: I am trying to answer to OpenAI assistants function tool. “我正在尝试回答 OpenAI 助手的函数工具。” See my code below. The custom function is called "funnyfunc&qu…

系统编程 网络 基于tcp协议

tcp的客户端: socket();用来开链接的端口 bind();绑定作用(在客户端可选可不选) connect();链接作用 tcp的服务端: s…

性能测试全解

世界上没有陌生人,只有还没认识的朋友 一.性能测试的意义 由于软件系统的性能问题而引起严重后果的事件比比皆是,下面列举几个案例 (1)2007年10月,北京奥组委实行2008年奥运会门票预售,一时间订票官网访问量激致系统…

「知识篇」UWB精确测距与定位技术优势的详细探讨

UWB650模块是思为无线新发布的一款双边双向测距,三点平面定位模块,WB650模块是在UWB3000F27基础上研发,并搭载单片机,用户无需配置可直接使用。 遵循IEEE 802.15.4-2020标准的UWB技术及其通信协议,提供高精度、低功耗…

第12章 网络 (1)

目录 12.1 互联的计算机 12.2 ISO/OSI 和TCP/IP 参考模型 12.3 通过套接字通信 12.3.1 创建套接字 12.3.2 使用套接字 12.3.3 UDP套接字 12.4 网络实现的分层模型 本专栏文章将有70篇左右,欢迎关注,查看后续文章。 网络相关的头文件数目巨大&…

两台电脑之间记事本内容如何转移?

记事本是我们日常生活中不可或缺的工具,它轻便、简单,方便我们随时记录生活中的点滴、工作中的灵感或重要的事务。比如,在会议中快速记下关键点,或者在阅读时捕捉一闪而过的想法。然而,随着数字化生活的推进&#xff0…

重塑“我店”平台:绿色积分引领的数字消费新纪元

在数字化转型的洪流中,“我店”平台凭借其创新的绿色积分体系异军突起,成为市场中的璀璨新星。本文将深度剖析“我店”的运营模式、市场效应及其如何通过绿色积分机制开创消费新潮流。 一、崛起之路与市场震撼力 自2021年盛夏在上海启航以来&#xff0c…

研讨会邀请函-Parasoft TÜV Rheinland|SOA架构下符合功能安全要求的软件自动化测试解决方案

尊敬的技术先锋, 在汽车行业的数字化转型浪潮中,软件安全已成为我们共同关注的焦点。Parasoft 联合 TV Rheinland,荣幸地邀请您参与我们即将举办的专业研讨会,与行业领袖一同探索SOA架构下的功能安全软件开发测试方案。 会议议程…

支付宝小程序websocket长连接(心跳版本)

注意点: 关闭连接一定要把那些开下来的监听全部关闭掉 1.开启连接 /*长连接*/ connectWebSocket() {let that this;my.connectSocket({url: ws://192.xx.8.xx:7780/charger-service-netty/websocket/${uni.getStorageSync(chargePointId)},header: {AccessType: a…

三种相机模型总结(针孔、鱼眼、全景)

相机标定 文章目录 相机标定前言 前言 我们最常见的投影模型Perspective Projection Model描述的就是针孔相机的成像原理。从上面的图根据相似三角形可以得出 参考链接 https://zhuanlan.zhihu.com/p/540969207 相机标定之张正友标定法数学原理详解(含python源码&a…

楼宇智慧公厕系统实时卫生状况一目了然

在科技飞速发展的今日,楼宇智慧公厕系统如一颗璀璨的新星,悄然改变着我们的生活。它以先进的技术手段,让公厕的实时卫生状况一目了然,为人们带来了全新的如厕体验。 当我们步入一栋现代化的楼宇,对公厕的期待不再仅仅是…

JVM 内存结构了解吗,每个区域都存放什么数据?

Java 程序是运行在 JVM 之中的,所有对象的创建和分配都在 JVM 中。 内存结构: 方法区:各线程共享,主要存放类信息、常量、静态变量 虚拟机栈:线程私有,主要存放基本数据类型(int、char、float……

Blazor开发框架Known-V2.0.9

V2.0.9 Known是基于Blazor的企业级快速开发框架,低代码,跨平台,开箱即用,一处代码,多处运行。本次版本主要是修复一些BUG和表格页面功能增强。 官网:http://known.pumantech.comGitee: https:…

什么是蒙太奇谎言

蒙太奇谎言,可以理解为不表述全部事实,而是进表达部分事实,让听众形成错误的观点。 比如,某X国家队水平很差,从来没进入过世界杯。 可以这样说:世界足球强国巴西,从来没在世界大赛上赢过X国家队…

C++构造数据类型|枚举类型

C构造数据类型|枚举类型 1. 枚举类型1.1 函数重载的定义1.2 枚举类型的声明1.3 例1:1.4 例2: 2. 枚举类型的定义说明3. 枚举类型的使用3.1 枚举变量的赋值3.2 枚举变量的运算3.3 枚举变量的输入3.4 注意事项 4 示例代码 1. 枚举类型 1.1 函数重载的定义…

cdr工具介绍之刻刀工具

在日常的生活当中,在很多时候我们会遇到各种各样的难题,但软件cdr他就是一个神奇的存在,因为他能帮助我们解决很多专业方面的的知识。尽管他的内容相比较其他的一些设计软件而言相对于较为少,但是他确实一个非常适合于平常的工作学…

arthas源码刨析:arthas-core (2)

文章目录 attach JVMagent**ArthasBootstrap** arthas-core的启动可以从上一篇做参考 参考 pom,即启动是调用的 Arthas 的 main 方法 attach JVM JVM提供了 Java Attach 功能,能够让客户端与目标JVM进行通讯从而获取JVM运行时的数据,甚至可以…