JavaScript递归技巧的前世今生:深入解析递归及其与堆栈的关系

🧑‍🎓 个人主页:《爱蹦跶的大A阿》

🔥当前正在更新专栏:《VUE》 、《JavaScript保姆级教程》、《krpano》

​ 

✨ 前言

        递归作为一种能够用简洁的方式定义复杂对象的编程技巧,在计算机科学中被广泛应用。它借助系统堆栈的先入后出结构,将大问题拆分为小问题来解决,对于二叉树、组合问题等都是非常高效的解决方案。

        但是递归也有其局限性,它占用堆栈空间,存在最大调用层数限制。为了发挥递归技巧的优势,同时规避其缺点,我们需要深入理解递归的原理及其与堆栈的关系,掌握改写递归的技巧。

        本文将从递归和堆栈的基本概念出发,剖析两者之间的内在联系,分析递归技巧的优缺点,给出其典型应用场景,并通过示例讲解递归代码的实现思路。最后,将提供改写递归的实用技巧,帮助读者更好地运用这一编程工具。

✨ 正文

一、递归和堆栈的概念

递归是一种编程技巧,它会直接或间接地调用函数自身来解决问题,就像一面镜子可以不断产生镜中的镜像一样。递归会把大问题分解为小问题来解决,小问题的解决方案又依赖于大问题的解决。

堆栈是计算机管理函数调用的一种数据结构,它采用先进后出的方式存储信息。当一个函数开始执行时,会被添加到堆栈的顶部;当一个函数执行结束,就会被堆栈删除。递归函数的实现就是利用了这种堆栈结构。

二、递归的两个必要条件

一个递归函数必须同时满足以下两个条件:

  1. 必须有一个基准条件(base case)。当满足这个条件时,递归不再继续。
  2. 递归条件(recursive case)。这一步把原问题分解成更小的子问题。

只要同时满足这两个条件,就可以使用递归来解决问题。

三、递归和堆栈的关系

递归函数在执行时会保存每一层的函数调用现场信息,这些信息被保存在系统的运行时堆栈中。可以把递归看成是不断把函数调用现场信息压入堆栈,并在满足基准条件时逐层将堆栈弹出并恢复函数的执行现场。

四、递归的优缺点

优点:

  1. 递归容易理解和实现。
  2. 递归可以用于解决分治法的问题。
  3. 递归允许程序以相对简洁的方式定义复杂对象。

缺点:

  1. 递归函数调用开销较大,占用堆栈空间。
  2. 递归超过最大调用层数会发生“栈溢出”错误。
  3. 递归函数调试困难。

五、递归的常见场景

  • 各种数学问题,如阶乘、斐波那契数列等。

(1)计算阶乘

function factorial(n) {if (n == 1) return 1;return n * factorial(n - 1);
}

(2)斐波那契数列 

function fib(n) {if (n <= 2) return 1;return fib(n-1) + fib(n-2);
}
  • 树和图的遍历。

以二叉树前序遍历为例:

function preorder(root) {if (!root) return;  console.log(root.val);preorder(root.left);preorder(root.right);
}
  • 动态规划灌水法。

比如实现DFS深度优先遍历:

function dfs(graph, current, visited) {visited.add(current);for (let neighbor of graph[current]) {if (!visited.has(neighbor)) {dfs(graph, neighbor, visited);}}
}
  • 逆序、分治、回溯等算法。

例如归并排序:

function mergeSort(arr) {if (arr.length === 1) {return arr;}const mid = Math.floor(arr.length / 2);const left = arr.slice(0, mid);const right = arr.slice(mid);return merge(mergeSort(left), mergeSort(right)); 
}

六、递归实现的案例

以计算阶乘为例,递归代码如下:

function factorial(n) {if (n === 1) {return 1}return n * factorial(n - 1)
}factorial(5) // 120

满足基准条件(n === 1)时返回1,否则不断调用自身计算n*(n-1)的阶乘。

七、递归的改写技巧

  1. 引入缓存,避免重复计算。
  2. 尝试用循环重写递归。
  3. 将高开销的递归改为低开销的迭代形式。
  4. 在尾递归优化的语言中使用尾递归优化。

 

✨ 结语

        理解递归技巧的本质在于掌握递归与堆栈的关系。递归允许程序以简洁的方式定义复杂对象,是计算机科学中一把双刃剑。运用得当,可以事半功倍;使用不当,后果不堪设想。

        本文全面解析了递归及其与堆栈的千丝万缕的联系,剖析了递归技巧的优劣势,并提供了改写递归的有效方法。希望这些知识可以帮助读者在运用递归技巧时既发挥它的高效优势,又规避其潜在缺陷。 

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

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

相关文章

OpenHarmony南向之LCD显示屏

OpenHarmony南向之LCD显示屏 概述 LCD&#xff08;Liquid Crystal Display&#xff09;驱动&#xff0c;通过对显示器上下电、初始化显示器驱动IC&#xff08;Integrated Circuit&#xff09;内部寄存器等操作&#xff0c;使其可以正常工作。 HDF Display驱动模型 LCD器件驱…

示例说明 Makefile 中的 $(@F),及其用法示例$$dir $@ $< $^ %.c

备忘一个不错的开源编辑器CudaText 下载网址&#xff1a; CudaText - Browse /release at SourceForge.net CudaText 主页&#xff1a; CudaText - Home 1&#xff0c;含义及验证 在 Makefile 中&#xff0c;$(F) 表示当前规则的目标文件名&#xff08;不包括路径部分&…

都2024年了,FP卖家还不知道AB站怎么玩?

自从开始写FP独立站各种运营技巧和黑科技的文章&#xff0c;经常都会有朋友来私V&#xff0c;询问怎么进行AB站跳转。 可能是现在平台对于FP商家的限制越来越多&#xff0c;再加上如今到处都是“内卷”的电商环境&#xff0c;让FP商家生存越来越艰难&#xff0c;今天就着重讲一…

【踩坑】JDK1.8 AudioSystem 无法关闭流的问题

文章目录 一、前言二、开始狼人杀嫌疑人1&#xff1a;嫌疑人2&#xff1a; 三、复盘Jdk8原生bug解决方法和原理解析 一、前言 做了一个基于文字转语言的小接口&#xff0c;想删除本地wav文件来着&#xff0c;结果发现删除不了。 很明显被占用了&#xff0c;还是被Java占用了……

Linux内核源码分析(强烈推荐收藏!)

前言&#xff1a;Linux内核是由林纳斯托瓦兹&#xff08;Linus Torvalds&#xff09;在1991年开始开发的。当时他为了得到一个可以运行UNIX操作系统的个人计算机&#xff0c;开始编写一个操作系统内核&#xff0c;并将其命名为Linux。随后&#xff0c;越来越多的开发者加入到项…

【论文】:ALOHA双手远程操作手臂

对于机器人来说&#xff0c;诸如穿扎带或插入电池等精细操作任务是众所周知的困难&#xff0c;因为它们需要精确度、接触力的仔细协调以及闭环视觉反馈。执行这些任务通常需要高端机器人、精确的传感器或仔细的校准&#xff0c;这些可能既昂贵又难以设置。学习能否使用低成本且…

计算机组成原理之计算机硬件发展和计算机系统的组成

学习的最大理由是想摆脱平庸&#xff0c;早一天就多一份人生的精彩&#xff1b;迟一天就多一天平庸的困扰。各位小伙伴&#xff0c;如果您&#xff1a; 想系统/深入学习某技术知识点… 一个人摸索学习很难坚持&#xff0c;想组团高效学习… 想写博客但无从下手&#xff0c;急需…

算法第4版 第2章排序

综述&#xff1a;5个小节&#xff0c;四种排序应用&#xff0c;初级排序、归并排序、快速排序、优先队列 2.1.初级排序 排序算法模板&#xff0c;less(), exch(), 排序代码在sort()方法中&#xff1b; 选择排序&#xff1a;如升序排列&#xff0c;1.找到数组中最小的元素&am…

llama_index 创始人为我们展示召回提升策略(提升15%)

用句子向量替换为句子向量 句子检索&#xff0c;将句子转化为向量。在检索的过程中&#xff0c;假如句子命中&#xff0c;则将句子周围的内容也当做检索内容。 对比句子检索和之前的按块去做切分的检索。可以看到&#xff0c;内容的相关性提升了8%&#xff0c; 构建数据的时候…

如何利用ChatGPT快速生成月报?

随着每个月的结束&#xff0c;个人和团队经常需要编写月报来回顾和总结。这项任务通常消耗大量时间和精力。幸运的是&#xff0c;借助ChatGPT&#xff0c;这个过程可以变得更加简单和高效。接下来&#xff0c;我将详细介绍如何利用ChatGPT快速生成月报&#xff0c;从而帮助你节…

NSSCTF Interesting_include

开启环境: 通过审计,我们可知: flag在flag.php中,可以利用php中伪协议 payload:?filterphp://filter/readconvert.base64-encode/resourceflag.php 将其base64解码就是flag. NSSCTF{3dc54721-be9e-444c-8228-7133fba76ad4}

大数据系列之:腾讯云服务器性能和价格比较

大数据系列之&#xff1a;腾讯云服务器性能和价格比较 一、磁盘性能和价格比较二、高性能云硬盘三、ssd云硬盘四、极速型ssd云硬盘五、增强型ssd云硬盘六、查看腾讯云服务器价格 一、磁盘性能和价格比较 磁盘名称高性能ssd云硬盘极速型ssd云硬盘增强型ssd云硬盘规格500g 5800 …

Navicat 技术干货 | 为 MySQL 表选择合适的存储引擎

MySQL 是最受欢迎的关系型数据库管理系统之一&#xff0c;提供了不同的存储引擎&#xff0c;每种存储引擎都旨在满足特定的需求和用例。在优化数据库和确保数据完整性方面&#xff0c;选择合适的存储引擎是至关重要的。今天&#xff0c;我们将探讨为 MySQL 表选择合适的存储引擎…

【BetterBench】2024年都有哪些数学建模竞赛和大数据竞赛?

2024年每个月有哪些竞赛&#xff1f; 2024年32个数学建模和数据挖掘竞赛重磅来袭&#xff01;&#xff01;&#xff01; 2024年数学建模和数学挖掘竞赛时间目录汇总 一月 &#xff08;1&#xff09;2024年第二届“华数杯”国际大学生数学建模竞赛 报名时间&#xff1a;即日起…

使用组合框QComboBox模拟购物车

1.组合框: QComboBox 组合框&#xff1a;QComboBox 用于存放一些列表项 实例化 //实例化QComboBox* comboBox new QComboBox(this);1.1 代码实现 1.1.1 组合框的基本函数 QComboBox dialog.cpp #include "dialog.h" #include "ui_dialog.h"Dialog::Dialog…

AC/DC控制电路选型分析

AC/DC控制电路选型&#xff0c;输出功率5W~20W&#xff0c;工作频率50KHz~100KHz UVL0/OVP/SCP/OCP/OLP等多种保护功能可选

C++ OpenGL 3D Game Tutorial 2: Making OpenGL 3D Engine学习笔记

视频地址https://www.youtube.com/watch?vPH5kH8h82L8&listPLv8DnRaQOs5-MR-zbP1QUdq5FL0FWqVzg&index3 一、main类 接上一篇内容&#xff0c;main.cpp的内容增加了一些代码&#xff0c;显得严谨一些&#xff1a; #include<OGL3D/Game/OGame.h> #include<i…

重新认识Elasticsearch-一体化矢量搜索引擎

前言 2023 哪个网络词最热&#xff1f;我投“生成式人工智能”一票。过去一年大家都在拥抱大模型&#xff0c;所有的行业都在做自己的大模型。就像冬日里不来件美拉德色系的服饰就会跟不上时代一样。这不前段时间接入JES&#xff0c;用上好久为碰的RestHighLevelClient包。心血…

模拟超市商品结算系统

要求:全程一个角色(管理员即用户) (1)需要管理员注册与登录 (2)管理员登录之后&#xff0c;可以进行上架新的商品(商品名称和单价) (3)管理员登录之后&#xff0c;也可以下架商品 (4)在节假日有优惠活动,可以对其中的一些商品修改相应的单价(价格提高和价格降低都可以) (5)用户…

如何使用CentOS系统中的Apache服务器提供静态HTTP服务

在CentOS系统中&#xff0c;Apache服务器是一个常用的Web服务器软件&#xff0c;它可以高效地提供静态HTTP服务。以下是在CentOS中使用Apache提供静态HTTP服务的步骤&#xff1a; 1. 安装Apache服务器 首先&#xff0c;您需要确保已安装Apache服务器。可以使用以下命令安装Ap…