闭包是函数与其词法作用域的组合,使内部函数能访问并“记住”定义时外层作用域的变量;它支撑数据封装、私有状态、异步回调,并需注意内存泄漏风险。
闭包是 JavaScript 中一个核心但常被误解的概念:它指的是**函数与其词法作用域的组合**,也就是说,一个函数不仅能访问自己作用域内的变量,还能“记住”并访问其定义时所处的外层作用域中的变量,即使外层函数已经执行完毕。
JavaScript 中的函数是一等公民,可以作为值被返回、传递或赋值。当一个内部函数在定义它的外部函数返回后,仍能访问外部函数的局部变量,就形成了闭包。
例如:
function createCounter() {这里 createCounter 返回的匿名函数就是一个闭包——它保留了对 count 的引用,而 count 并非全局变量,也不在调用时的作用域中,却始终可被访问和修改。
JavaScript 在 ES6 之前没有真
正的私有字段语法,闭包是实现模块化和信息隐藏的重要手段。
在事件处理、定时器、Promise 回调、fetch 请求等异步操作中,回调函数往往需要访问定义时的上下文变量。
例如:
for (var i = 0; i setTimeout(() => console.log(i), 100);这是因为 var 声明的 i 是函数作用域,所有回调共享同一个 i;若改用 let 或闭包可修复:
for (var i = 0; i (function(n) {这个立即执行函数创建了一个闭包,把每次循环的 i 值“捕获”为参数 n,从而保留了正确状态。
闭包会阻止外部作用域中变量被垃圾回收,如果持有大量数据或 DOM 引用,可能引发内存泄漏。