闭包是函数与其词法作用域的组合,能访问并保持外部变量的引用。如 inner 函数通过闭包使 outer 的 count 在外部长久存在,实现私有变量、模块封装等功能。

JavaScript闭包是开发中非常核心的概念,理解它不仅能帮助你写出更高效的代码,还能让你真正掌握函数式编程的精髓。闭包的本质是函数能够“记住”其定义时所处的环境,即使这个函数在其他地方被调用,也能访问到当时的变量。
闭包的基本原理
当一个内部函数引用了外部函数的变量时,就形成了闭包。这个内部函数会保留对外部作用域的引用,使得外部函数的变量不会被垃圾回收机制清除。
看一个简单例子:
function outer() {let count = 0;
return function inner() {
count++;
console.log(count);
};
}
const counter = outer();
counter(); // 1
counter(); // 2
这里 inner 函数形成了闭包,它持有了对 count 变量的引用。即使 outer 执行完毕,count 依然存在,因为闭包在使用它。
立即学习“Java免费学习笔记(深入)”;
闭包的常见应用场景
闭包不只是理论,它在实际开发中有很多高级用途:
- 私有变量模拟:JavaScript没有原生的私有属性,但可以用闭包实现数据隐藏。 function createBankAccount(initial) {
- 事件回调与定时器:在异步操作中,闭包常用于保存上下文信息。 for (var i = 1; i setTimeout(() => console.log(i), 100); // 输出 4, 4, 4
- 函数柯里化(Currying):利用闭包实现参数的逐步传递。 function add(a) {
let balance = initial;
return {
deposit: function(amount) { balance += amount; },
withdraw: function(amount) { if (amount getBalance: function() { return balance; }
};
}
const acc = createBankAccount(100);
acc.getBalance(); // 100
balance 无法从外部直接访问,只能通过返回的对象方法操作。
}
由于 var 没有块级作用域,所有回调共享同一个 i。使用闭包可以解决这个问题:
for (let i = 1; i setTimeout(() => console.log(i), 100); // 输出 1, 2, 3}
或者用 IIFE(立即执行函数)手动创建闭包:
for (var i = 1; i (function(num) {setTimeout(() => console.log(num), 100);
})(i);
}
return function(b) {
return a + b;
};
}
add(2)(3); // 5
第一次调用保存了 a 的值,第二次调用使用闭包访问 a 并加上 b。
闭包的潜在问题与优化
虽然闭包功能强大,但使用不当也会带来问题:
- 内存泄漏风险:闭包会阻止变量释放,如果引用大型对象且不再需要,应手动解除引用。
- 性能考虑:频繁创建闭包可能影响性能,尤其是在循环中。
- 变量共享陷阱:多个闭包共享同一个外部变量时,修改会影响所有闭包。
避免这些问题的关键是明确变量生命周期,合理设计作用域结构。
基本上就这些。闭包不是魔法,它是作用域链和词法环境自然的结果。掌握它,你的 JavaScript 水平会上一个台阶。不复杂但容易忽略。










