0

0

什么是javascript递归函数_如何避免递归导致栈溢出【教程】

狼影

狼影

发布时间:2026-01-19 18:07:02

|

467人浏览过

|

来源于php中文网

原创

递归函数是直接或间接调用自身的函数,常用于树遍历、阶乘等场景;关键是有自身调用且需终止条件,否则易因调用栈溢出导致rangeerror。

什么是javascript递归函数_如何避免递归导致栈溢出【教程】

递归函数是什么:调用自身的函数

JavaScript 递归函数就是函数在执行过程中,直接或间接调用自己。它不是语法特性,而是一种编程模式,常用于遍历树结构、计算阶乘、解析嵌套对象等场景。

关键判断标准:function foo() { foo(); } —— 只要函数体内有对自身的调用,且没有可靠的终止条件,就构成递归。

但递归本身不危险,危险的是「无限递归」或「深度过大」。JS 引擎对调用深度有限制(通常 10k–15k 层,具体取决于引擎和内存),超过即抛出 RangeError: Maximum call stack size exceeded

为什么递归容易栈溢出:每次调用都压栈

JS 是单线程、基于调用栈执行的。每次函数调用,引擎都会在栈上保存当前执行上下文(变量、参数、返回地址)。递归每深入一层,就新增一帧;返回时才逐层弹出。

立即学习Java免费学习笔记(深入)”;

智川X-Agent
智川X-Agent

中科闻歌推出的一站式AI智能体开发平台

下载

这意味着:

  • factorial(10000) 可能成功,factorial(20000) 很可能失败
  • 深度为 100 的树遍历,若用朴素递归,栈深就是 100 —— 看似安全,但若树不平衡(比如退化成链表),实际可能逼近极限
  • 箭头函数、async 函数、generator 中递归,同样受栈限制,不会自动优化

如何避免栈溢出:改写策略与边界控制

不能一概禁用递归,而是根据场景选择更稳妥的实现方式。核心思路是:把「依赖调用栈维护状态」转为「手动管理状态」。

推荐做法:

  • 对已知深度较大的场景(如扁平化 1000+ 层嵌套数组),优先用 while + stack 模拟递归,例如:
    function flattenDeep(arr) {
      const result = [];
      const stack = [...arr];
      while (stack.length > 0) {
        const val = stack.pop();
        if (Array.isArray(val)) {
          stack.push(...val);
        } else {
          result.push(val);
        }
      }
      return result.reverse();
    }
  • 对可预测的浅层递归(如 DOM 树遍历、JSON Schema 验证),加深度计数器强制中断:
    function traverse(node, depth = 0, maxDepth = 100) {
      if (depth > maxDepth) throw new Error('Recursion depth exceeded');
      // 处理 node...
      for (const child of node.children || []) {
        traverse(child, depth + 1, maxDepth);
      }
    }
  • 尾递归优化(TCO)理论上可解,但目前所有主流浏览器都不支持 ES6 尾调用优化,写成尾递归形式(return fn(...))也无济于事,栈照样增长

容易被忽略的隐式递归点

有些代码看似没递归,实则触发了隐蔽的调用链:

  • JSON.stringify() 序列化循环引用对象时,会报 TypeError: Converting circular structure to JSON,但若自定义 toJSON 方法又调回自身,就变成真递归
  • Vue / React 中,computeduseMemo 依赖项里读取另一个计算属性,而后者又反向依赖前者 → 无限求值,本质是逻辑递归
  • 事件监听器中触发相同事件(如 input 改值 → dispatchEvent → 又触发 input),形成事件递归,虽不占 JS 调用栈,但会导致 UI 卡死或内存暴涨

真正难调试的,往往是这些非显式、跨模块、带异步延迟的递归路径。加日志时别只打 console.log('enter'),带上 new Error().stack.split('\n')[1] 快速定位调用源头。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

455

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

546

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

334

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

82

2025.09.10

es6新特性
es6新特性

es6新特性有:1、块级作用域变量;2、箭头函数;3、模板字符串;4、解构赋值;5、默认参数;6、 扩展运算符;7、 类和继承;8、Promise。本专题为大家提供es6新特性的相关的文章、下载、课程内容,供大家免费下载体验。

106

2023.07.17

es6新特性有哪些
es6新特性有哪些

es6的新特性有:1、块级作用域;2、箭头函数;3、解构赋值;4、默认参数;5、扩展运算符;6、模板字符串;7、类和模块;8、迭代器和生成器;9、Promise对象;10、模块化导入和导出等等。本专题为大家提供es6新特性的相关的文章、下载、课程内容,供大家免费下载体验。

197

2023.08.04

JavaScript ES6新特性
JavaScript ES6新特性

ES6是JavaScript的根本性升级,引入let/const实现块级作用域、箭头函数解决this绑定问题、解构赋值与模板字符串简化数据处理、对象简写与模块化提升代码可读性与组织性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

232

2025.12.24

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

106

2023.09.25

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Vue 教程
Vue 教程

共42课时 | 9.5万人学习

Vue3.x 工具篇--十天技能课堂
Vue3.x 工具篇--十天技能课堂

共26课时 | 1.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号