闭包是函数作用域链自然保留的结果:外层函数返回内层函数,内层函数引用外层变量,且外层执行完后内层仍可调用;如createCounter中count被持续持有,而foo中bar立即调用则不构成闭包。

闭包不是某种特殊语法,而是 JavaScript 中函数作用域链自然保留的结果:当一个函数返回了内部定义的函数,并且这个内部函数在外部被调用时,它仍能访问自己定义时所在作用域里的变量——这就形成了闭包。
闭包形成的三个必要条件
闭包不是写出来的,是“跑出来”的。只要同时满足以下三点,闭包就存在:
- 外层函数定义并
return了一个内层函数(或以其他方式暴露出去,比如赋值给全局变量、传入回调等) - 内层函数中引用了外层函数的局部变量(
let、const或var声明的) - 外层函数执行完毕后,内层函数依然可被调用(即没有被垃圾回收)
function createCounter() 是最典型的闭包示例
它直观展示了变量如何“活”在函数调用结束后:
function createCounter() {
let count = 0;
return function() {
count++;
return count;
};
}
const counter = createCounter();
console.log(counter()); // 1
console.log(counter()); // 2
这里 count 不是全局变量,也不是参数传进来的,它被 createCounter 的执行上下文创建,又被返回的匿名函数持续持有——JavaScript 引擎必须保留这个上下文,否则 count 就会丢失。
立即学习“Java免费学习笔记(深入)”;
常见误判:不是所有嵌套函数都是闭包
闭包的关键在于「外部调用时仍能访问原作用域」。下面这段代码不构成闭包:
function foo() {
const x = 42;
function bar() {
console.log(x); // 引用了外层变量
}
bar(); // 在 foo 内部立即调用
}
foo();
因为 bar 没有逃出 foo 的作用域,foo 执行完后整个上下文可被回收,x 不会被长期持有。
真正容易忽略的是:闭包持有的是变量的「引用」,不是值。多个闭包共享同一外层变量时,修改会互相影响——这点在循环中创建函数时特别容易出错。











