闭包是函数与其词法作用域的组合,使内部函数能持续访问外部函数执行后本该销毁的变量;典型问题如setTimeout中循环索引值错误,可通过IIFE、let声明或箭头函数结合闭包机制解决。

闭包不是语法糖,而是函数与其词法作用域的组合;它让内部函数能持续访问外部函数执行完后本该销毁的变量。
为什么 setTimeout 里取不到循环中的正确索引值?
这是闭包最典型的应用场景,也是新手最容易卡住的地方。问题本质是:循环快速结束,i 最终变成 3,而所有回调共享同一个 i 的引用。
- 错误写法:
for (var i = 0; i < 3; i++) { setTimeout(() => console.log(i), 100); }输出全是3 - 用闭包修复:把当前
i封进一个立即执行函数,形成独立作用域for (var i = 0; i < 3; i++) { (function(j) { setTimeout(() => console.log(j), 100); })(i); } - ES6 更简洁:用
let声明,每次迭代绑定新绑定,本质也是靠块级作用域+闭包机制支撑
return 出来的函数为什么还能读写外层变量?
只要函数在定义时处于某个作用域内,且被返回或传递出去,它就“记住”了那个作用域的变量环境——哪怕外层函数早已执行完毕、栈帧已弹出。
-
var counter = (function() { let count = 0; return () => ++count; })();中,count不会释放,因为匿名函数形成了闭包 - 多次调用
counter()得到递增结果,说明count是私有且持久的 - 注意:过度使用闭包会阻止垃圾回收,尤其在 DOM 事件监听器中持有大对象时容易内存泄漏
闭包和 this 绑定冲突怎么办?
闭包捕获的是词法作用域里的变量,但 this 是动态绑定的,两者不自动同步。常见于事件回调或定时器中 this 指向丢失。
立即学习“Java免费学习笔记(深入)”;
- 错误示例:
const obj = { value: 42, init() { setTimeout(function() { console.log(this.value); }, 100); } };输出undefined(this指向全局) - 用闭包保存
this:init() { const self = this; setTimeout(function() { console.log(self.value); }, 100); } - 更现代写法:箭头函数不绑定
this,直接继承外层词法this,本质依赖闭包机制init() { setTimeout(() => console.log(this.value), 100); }
真正难的不是写出闭包,而是判断变量是否真的被闭包持有了——比如在循环中创建大量函数却忘了清理引用,或者误以为 const 声明就能避免闭包持有对象。调试时多看 console.dir(fn) 里的 [[Scopes]],那里藏着所有被闭包捕获的变量。











