当前位置: 首页 > news >正文

【JavaScript】递归的问题以及优化方法

1. 递归多会导致内存消耗变大

  • 递归的本质:函数调用自己,每次调用会在 调用栈 上压入一帧(栈帧),里面保存:

    • 局部变量
    • 参数
    • 返回地址
    • 执行上下文
  • 当递归层级很多时:

    • 调用栈帧越来越多 → 内存消耗增加
    • 如果层级过深 → 触发 栈溢出(Stack Overflow)

👉 举例:

function factorial(n) {if (n === 1) return 1;return n * factorial(n - 1);
}
factorial(100000); // 🔥 Maximum call stack size exceeded

这里就是因为递归深度太大,每一层都占内存,最后栈爆了。


2. 怎么优化递归?

(1)尾递归优化(Tail Recursion)

  • 原理:如果递归调用是函数的最后一步,可以不需要保留当前栈帧,直接复用(更新参数,而不是创建新的栈帧) → 内存消耗低。
  • ES6 有尾调用优化的提案,但目前大部分 JS 引擎 没实现(在某些语言如 Scala、Haskell 是默认优化)。
// 普通递归
function sum(n) {if (n === 1) return 1;return n + sum(n - 1);
}// 尾递归
function sumTail(n, acc = 0) {if (n === 0) return acc;return sumTail(n - 1, acc + n); // ✅ 最后一步是递归调用
}sumTail(100000, 0); // 理论上不会栈溢出

(2)递归改迭代

  • 思路:用循环 + 栈/队列模拟递归,避免调用栈过深。
// 递归版
function factorial(n) {if (n === 1) return 1;return n * factorial(n - 1);
}// 迭代版
function factorialIter(n) {let res = 1;for (let i = 1; i <= n; i++) {res *= i;}return res;
}

👉 迭代的空间复杂度通常是 O(1),比递归更省内存。


(3)分治 + 记忆化(减少重复递归)

  • 比如 斐波那契数列,递归会大量重复计算,可以用缓存减少递归次数。
function fib(n, memo = {}) {if (n <= 1) return n;if (memo[n]) return memo[n];memo[n] = fib(n - 1, memo) + fib(n - 2, memo);return memo[n];
}

3. 面试总结答法

递归会导致内存消耗变大,是因为每次函数调用都会在调用栈里保存上下文,如果递归层级太深,栈就会溢出。优化方法包括:

  • 使用 尾递归优化(如果语言/引擎支持)。
  • 将递归改写为 迭代,降低空间复杂度。
  • 使用 记忆化缓存,减少重复递归。
http://www.dtcms.com/a/355865.html

相关文章:

  • week5-[一维数组]去重
  • (笔记)Android窗口管理系统分析
  • 向量方法证明正余弦定理的数学理论体系
  • 如何保证数据的安全性和隐私性?
  • Spring Boot + KingbaseES 连接池实战
  • TypeScript:枚举类型
  • Milvus向量数据库是什么?
  • Active Directory Basics
  • UPAM(Unified Prompt Attack Model
  • 应急响应/windows权限维持/Linux权限维持
  • 虚拟机逃逸攻防演练:从攻击模拟到隔离漏洞防御实战
  • 机器学习回顾(二)——KNN算法
  • 【Cadence技巧】立创EDA/Altium/Allegro之间的封装转换
  • layout版图设计学习笔记2_工艺流程
  • 切入高潜市场,抢占行业先机!ES SHOW 2025展位预订火爆,10月28-30日共启增长新蓝海
  • php姓名三才五格api接口调用说明
  • 疯狂星期四文案网第53天运营日记
  • gdbserver远程调试和交叉编译gdb
  • Fuzzy Multimodal Learning for Trusted Cross-modal Retrieval(CVPR 2025)
  • OpenCV 图像操作进阶:像素、边界与融合技术
  • 数据结构青铜到王者第九话---二叉树(2)
  • 多语言与零样本语音识别新突破:基于发音特征分类的方法
  • 通过ETL工具,同步SQLserver数据至starrocks数据库
  • Autosar之DCM模块
  • 构建AI智能体:十六、构建本地化AI应用:基于ModelScope与向量数据库的文本向量化
  • Day14 Gorm框架学习(1)
  • 安装与环境搭建:准备你的 Electron 开发环境
  • leetcode 525 连续数组
  • 可改善能源利用水平、削减碳排放总量,并为可再生能源规模化发展提供有力支撑的智慧能源开源了
  • 计算机组成原理3-3-5:定点数的乘法运算——补码阵列乘法器