数据结构-递归

用递归代替循环

假设工作中的你,需要写一个倒数程序。该程序接收一个数字,例如10,然后显示从10到0的数字。现在先暂停一下,选择一门编程语言来实现这个程序,做完以后,再往下阅读。或许你用了JavaScript,并且写了如下循环。

这样写没什么问题,只是你可能没想到循环以外的做法。那还能怎么做呢?试试换成递归吧。以下是初级版的递归countdown。

让我们一步步来分析。

第1步:调用countdown(10),因此参数number为10。

第2步:将number(值为10)打印到控制台。

第3步:countdown函数在结束前,调用了countdown(9)(因为number -1等于9)​。

第4步:countdown(9)被执行,会将number(值为9)打印到控制台。

第5步:countdown(9)结束前,调用了countdown(8)。

第6步:countdown(8)被执行,会将number(值为8)打印到控制台。

在继续步骤分解之前,先回顾下该递归是怎样实现我们的需求的。countdown里并没有任何循环结构,它通过调用自身就能够从10开始倒数并将每个数字打印出来。

几乎所有循环都能够转换成递归。但能用不代表该用。递归的强项在于巧妙地解决问题,但在上面的例子中,它并不比普通的循环更加优雅、高效。我们很快就会看到能让递归发挥威力的场景,但在那之前,还是先理清递归的运作方式。

基准情形

让我们把countdown函数继续下去。为了简洁一点,我们跳过一些步骤。

第21步:调用countdown(0)。

第22步:将number(值为0)打印到控制台。

第23步:调用countdown(-1)。

第24步:将number(值为-1)打印到控制台。你也看到了,这种写法不够完善,这样下去我们就会不断地打印负数。

要解决这个问题,得在数到0时就停住,以免递归一直往下数。

我们可以加个条件判断,来保证当number为0时,不再调用countdown()。

这样,当number为0时,我们的代码就不会再去调用countdown(),而是直接返回。

在递归领域(真有这么一个地方)​,不再递归的情形称为基准情形。对于刚才的countdown()函数来说,0就是基准情形。

阅读递归代码

递归是需要时间和练习才能适应的,到那时候,你会掌握两种技巧:阅读递归代码和编写递归代码。阅读递归代码相对简单一点,所以就先从这里入手吧。我们会以阶乘作为例子。阶乘的演示如下所示。3的阶乘是:

5的阶乘是:

以此类推。以下Ruby代码会以递归计算的方式返回一个数的阶乘。

此代码初看可能会让人有点困惑,可以按照以下流程来读。

(1) 找出基准情形。

(2) 看该函数在基准情形下会做什么。

(3) 看该函数在到达基准情形的前一步会做什么。

(4) 就这样往前推,看每一步都在做什么。

让我们将此流程应用到刚才的代码上。稍作分析,就可以看出里面有两条路径。

第二条路的factorial有调用自身,是递归发生的地方。

第一条路并没有调用自身,因此这里是基准情形。

于是,number为1时,是基准情形。接着,想象factorial方法在基准情形下,即factorial(1)的处理流程。其相关代码如下。

好,这很简单,因为是基准情形,所以没有递归。调用factorial(1)就会直接返回1。于是找来一张纸,记下该结果。

然后,回到上一步的factorial(2),相关代码如下。

调用factorial(2)就会返回2 * factorial(1)。要计算2 * factorial(1),就得先知道factorial(1)的结果。要是检查下前面所记,你会发现那是1。因此,2 * factorial(1)就是2 * 1,即是2。把这个也记到纸上。

那么,factorial(3)又会是什么呢?再回看代码。

代入参数便是3 * factorial(2)。那么factorial(2)是什么呢?你不用从头计算,因为它的结果已经写在纸上了,是2。于是factorial(3)会返回6(3 * 2=6)​。将结果记下,然后继续。

现在请自行计算factorial(4)。如你所见,这种从基准情形入手再往上分析的思路,对理解递归代码是多么有益。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数学建模--二分法

目录 二分法的基本原理 应用实例 求解方程根 查找有序数组中的元素 注意事项 Python代码示例 ​编辑 延伸 二分法在数学建模中的具体应用案例有哪些? 如何选择二分法的初始区间以确保收敛速度和精度? 在使用二分法求解方程时,如何…

排序算法2:直接选择排序与快速排序

目录 1.直接选择排序 1.1直接选择排序的优化 2.快速排序 2.1基准值的置位(Hoare版) 2.2挖坑法 2.3lomuto前后指针 前言 前面我们进入了排序算的讲解。今天我们将继续学习几种重要的排序思想,好,咱们三连上车开始今天的内容。…

ChatTTS文本转语音本地部署结合内网穿透实现远程使用生成AI音频

文章目录 前言1. 下载运行ChatTTS模型2. 安装Cpolar工具3. 实现公网访问4. 配置ChatTTS固定公网地址 前言 本篇文章主要介绍如何快速地在Windows系统电脑中本地部署ChatTTS开源文本转语音项目,并且我们还可以结合Cpolar内网穿透工具创建公网地址,随时随…

动态规划.

目录 (一)递归到动规的一般转化方法 (二)动规解题的一般思路 1. 将原问题分解为子问题 2. 确定状态 3. 确定一些初始状态(边界状态)的值 4. 确定状态转移方程 (三)能用动规解…

【网络】HTTP协议

目录 概述 URL 结构 urlencode(URL编码) urldecode(URL解码) 工具网址 HTTP请求 请求行 请求头 请求体 HTTP响应 状态行 响应头 响应体 个人主页:东洛的克莱斯韦克-CSDN博客 概述 HTTP协议是应用层协议…

TCP 三次握手建立连接

一开始,客户端和服务端都处于 CLOSE 状态。先是服务端主动监听某个端口,处于 LISTEN 状态 1. 第一次握手 客户端会随机初始化序号(client_isn),将此序号置于 TCP 首部的「序号」字段中,同时把 SYN 标志位置…

略读ArrayList源码

ArrayList是Java集合框架中的一部分,底层是通过数组实现的,可以动态增长和缩减。 一、首先看成员变量 序列化ID定义。在Java中,如果一个类实现了Serializable接口,那么它的serialVersionUID就非常重要了。serialVersionUID用于确…

python 图片爬虫记录

感谢大家的点赞。再补充一点。 对于这个 url https://blogger.googleusercontent.com/img/b/R29vZ2xl/AVvXsEjEqB5nighYsMZE7kexaVNJfxy3OkRutNEKatksw9u5f-ckHNROLzFyx2Uty3zYWNEaeOmzsljGr3eARiDWaM9DM8G2hPuPf8uZP0NO3kNUCnM2Cjb3ZKtLhJDBwqeR4ElpJ7ID5_wIHGQ/s200 这个url最…

Python进阶 JSON数据,pyecharts制图

目录 json数据格式的转换 什么是json json本质 注意 pyecharts快速入门 画一个最简单的折线图 使用全局配置选项优化折线图 总结 json数据格式的转换 什么是json 一种轻量级的数据交换格式,可以按json指定的格式去组织和封装数据 json本质 带有特定格式的…

汇川技术|Inoproshop基本使用方法:汇川指令库、库文件

哈喽,你好啊,我是雷工! 本节熟悉了解汇川常用指令库的分类及概述,了解Inoproshop库文件; 以下为学习笔记。 01 指令简介与分类 可编程控制系统中,使CPU完成某种操作或实现某种功能的命令及多个命令的组合…

CCRC-DSA数据安全评估师:加快构建大网络安全工作格局

7月31日,第十二届ISC.AI互联网安全大会开幕式在北京国家会议中心隆重举行,本次大会以“构建大型安全防护模型,引领安全产业创新”为主题。 中央网络安全和信息化委员会办公室副主任、国家互联网信息办公室副主任王京涛出席并发表了重要讲话。…

语音平台调研

百度DuerOS开放平台 DuerOS是百度推出的对话式人工智能操作系统,即智能语音交互平台。DuerOS的技术架构包含“对话服务”和“技能框架”两大基础协议。两大协议连通起来的对话核心系统、智能设备开放平台和技能开放平台,构成了完整DuerOS的智能生态系统。…

C#初级——字典Dictionary

字典 字典是C#中的一种集合&#xff0c;它存储键值对&#xff0c;并且每个键与一个值相关联。 创建字典 Dictionary<键的类型, 值的类型> 字典名字 new Dictionary<键的类型, 值的类型>(); Dictionary<int, string> dicStudent new Dictionary<int, str…

Javascript常见算法(二)【学习】

动态规划 斐波那契数列&#xff1a; 经典的动态规划问题&#xff0c;每个数是前两个数的和。 斐波那契数列&#xff08;Fibonacci sequence&#xff09;是一个非常著名的数列&#xff0c;其中每个数是前两个数的和&#xff0c;序列以0和1开始。在JavaScript中&#xff0c;有多…

药厂子母钟系统,强抗干扰能力,满足复杂生产环境

在制药行业中&#xff0c;精确的时间同步对于确保药品生产的质量和合规性至关重要。药厂子母钟系统作为一种高度可靠的时间同步解决方案&#xff0c;不仅能够提供准确的时间信息&#xff0c;还具有强大的抗干扰能力&#xff0c;非常适合在复杂的生产环境中使用。本文将详细介绍…

[STM32]HAL库实现自己的BootLoader-BootLoader与OTA-STM32CUBEMX

目录 一、前言 二、BootLoader 三、BootLoader的实现 四、APP程序 五、效果展示 六、拓展 一、前言 听到BootLoader大家一定很熟悉&#xff0c;在很多常见的系统中都会存在BootLoader。本文将介绍BootLoader的含义和简易实现&#xff0c;建议大家学习前掌握些原理基础。 …

YOLOV8替换Lion优化器

YOLOV8替换Lion优化器 1 优化器介绍博客 参考bilibili讲解视频 论文地址&#xff1a;https://arxiv.org/abs/2302.06675 代码地址&#xff1a;https://github.com/google/automl/blob/master/lion/lion_pytorch.py """PyTorch implementation of the Lion …

C++初学(11)

不知不觉就第11篇了QWQ 11.1、指针和自由存储空间 之前提到了计算机程序在存储数据时必须跟踪的3个基本属性&#xff1a; &#xff08;1&#xff09;信息存储在何处&#xff1b; &#xff08;2&#xff09;存储的值为多少&#xff1b; &#xff08;3&#xff09;存储的信息…

未授权访问漏洞(非重点 中)

6.Hadoop 1.在 fofa 使用 port"8088" && app"Hadoop" 获取资源 2.打开后若无需登录,则存在漏洞 7.ActiveMQ 1.在 fofa 使用 body"ActiveMQ" && port"8161" 获取资源 2.打开后若点击登录,默认账户密码为 admin/adm…

【css】使用CSS绘制奥运五环--巴黎奥运

使用CSS绘制奥运五环 在2024年巴黎奥运会期间&#xff0c;本文来使用 CSS 来画一个奥运五环。奥运五环由五个相互交叠的圆环组成&#xff0c;分别代表五大洲。 奥运五环是相互连接的&#xff0c;因此在视觉上会产生重叠效果&#xff0c;这也是实现五环最有挑战性的部分 HTML结…