javascript闭包是什么以及如何形成?【教程】

10次阅读

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

javascript闭包是什么以及如何形成?【教程】

闭包不是某种特殊语法,而是 javaScript 中函数作用域链自然保留的结果:当一个函数返回了内部定义的函数,并且这个内部函数在外部被调用时,它仍能访问自己定义时所在作用域里的变量——这就形成了闭包。

闭包形成的三个必要条件

闭包不是写出来的,是“跑出来”的。只要同时满足以下三点,闭包就存在:

  • 外层函数定义并 return 了一个内层函数(或以其他方式暴露出去,比如赋值给全局变量、传入回调等)
  • 内层函数中引用了外层函数的局部变量letconstvar 声明的)
  • 外层函数执行完毕后,内层函数依然可被调用(即没有被垃圾回收)

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 不会被长期持有。

真正容易忽略的是:闭包持有的是变量的「引用」,不是值。多个闭包共享同一外层变量时,修改会互相影响——这点在循环中创建函数时特别容易出错。

text=ZqhQzanResources