闭包是JavaScript引擎在函数嵌套、内部函数访问外部变量且该函数被传出作用域时自动创建的“函数+词法环境”绑定;它延长变量生命周期,可能导致内存泄漏。
闭包不是语法糖,也不是高级技巧——它是 JavaScript 引擎在满足特定条件时自动创建的一种「函数 + 词法环境」的绑定状态。它直接改写变量的生命周期,既能让 count 持久存在,也可能让一个 10MB 的 userProfile 对象永远卡在内存里出不去。
只有当以下三点同时成立,JavaScript 才会真正创建闭包:
inner() 定义在 outer() 内部inner 显式访问了 outer 的局部变量(如 let name = "Alice"),而不是只调用其他函数inner 被传出 outer 的作用域:作为返回值、赋给全局变量、传给 setTimeout 或 addEventListener
如果 inner 什么外部变量都不用,或者定义后立刻执行、没传出去,那它就只是普通嵌套函数,不构成闭包。
正常情况下,outer() 执行完,它的执行上下文弹出栈,count 理应被垃圾回收。但一旦 inner 形成闭包并持有对 count 的引用,引擎就必须保留这个变量所在的整个词法环境——哪怕你只用了其中 1 个字段,其余 9 个未使用的变量也跟着“陪绑”。
常见误判:
let 就能避免闭包问题 → 错。只要满足三条件,let 和 var 都会形成闭包,只是前者能正确捕获循环变量{ get: (
) => count }),照样闭包闭包本身不危险,危险的是它和 DOM、定时器、事件监听器混搭后形成的强引用链。
function attachHandler(element) {
const data = largeObject(); // 假设 5MB
element.addEventListener('click', () => {
console.log(data.id); // 只需要 id,却拖着整个 data
});
}
// 卸载组件时忘记 element.removeEventListener → data 永远不释放
实操建议:
data.id 替代整个 data 对象传入闭包clearTimeout、removeEventListener,或把闭包函数设为 null
WeakMap 关联私有数据:避免强引用阻碍回收,例如 const privateData = new WeakMap()
最常被忽略的一点:V8 确实会优化掉未使用的捕获变量,但这个“是否使用”完全由你的代码逻辑决定——引擎不会猜你要不要 data.config,它只看字节码里有没有读取指令。写的时候少引一个字段,内存里就少扛一份负担。