Javascript闭包真经
继前阵子写完对象真经后,这篇文章我尝试尽力的去讲透Js中的闭包。这里要感谢爱民,爱民的书写得很好,我从中获益良多。不过这次我打算换一种思路来写这篇真经,就是采用提问回答的方式,我下面先提出我要回答的问题,如果读者你都很自信的能够回答上,那么就可以考虑干别的事情去了。如果感觉自己有点把握不准就请给我一步步的寻址吧。:)我保证最后你就会豁然开朗,明白闭包的真谛。问题集:
什么是函数实例呢?其实在我们平常书写代码的过程中,写的函数就是一段文本,只是对于编译型语言来说会把它编译为确定的二进制代码,并放到确定的内存位置执行,而对于Js这样的解释型语言,就会在程序运行的时候把这段文本翻译为计算机能懂的话。那么这里函数代码的文本其实就是这个函数的类,而Js引擎解释后的内存中的数据就是这个函数的实例了。所以函数的实例就是Js引擎读过这段代码后在内存中产生的一段数据。
什么是函数引用呢?函数引用就是指向刚才所说的那段内存数据的指针。也就是指向某个函数实例的指针。同一份实例可以有多个引用,只要该实例还有指向它的引用,占用的内存就不会被释放。
什么是闭包呢?闭包就是函数实例执行过程中动态产生的一块新的内存里的数据集。这句话有可以表达两层意思:
- 最初闭包必须由函数实例被调用(也就是函数实例执行)时才会由Js引擎动态生成;
- 既然闭包也是一段内存区域,当没有依赖于这个内存中数据的引用时,才会被释放,也就是闭包理论上才会被销毁。
记住:闭包是执行期的概念!
那闭包里有什么玩意呢?这里我要引入Js引擎在语法分析期就构造的两类结构。执行上下文结构(Context)和调用对象结构(CallObject)。我们还是先看一幅图片:
从上面的图片我们容易看出,这是描述一个函数实例信息的图。Context结构包含了myFunc函数的类型(FUNCTION)、名称(myFunc)、形式参数(x,y,z)和最关键的CallObject结构的引用(body为CallObject)。而CallObject结构就是很详细的记录了这个函数的全部语法分析结构:内部声明的变量表(localDeclVars)、内部声明的具名函数表(localDeclFuncs)以及除去以上内容之外的其它所有代码文本的字符串表示(source)。
好,现在有了这个语法分析期就产生的CallObject结构,就好分析闭包里面会有什么了。当一个函数实例被调用时,实际上是从这个原型复制了一份CallObject结构到闭包那块空的内存中去。同时会给里面的数据赋值。这里有两个规则,大家要记下:
- 在函数实例开始执行时,localDeclVars中的所有值将被置为undefined;
- 在函数实例执行结束并退出时,localDeclVars不被重置,这就是Js中函数能够在内部保存数据的特性的原因;
- 这个闭包中的CallObject结构中的数据能保留多久取决于是否还有对其的引用存在,所以这里就引出了一个推论,闭包的生存周期不依赖于函数实例。
那么对于全局对象呢?是什么样的结构呢?其实和上面的函数实例的基本一样,唯一不同的是,没有Context,因为它在最顶层了。全局对象里面也有一个CallObject,只是这个CallObject有点特殊,就是localDeclVars里面的数据只会初始化一次,且整个CallObject里的数据总不被销毁。
除了我讲的复制了一份CallObject,再往里赋值,闭包其实还包括了一个upvalue数组,这里数组里装载的是闭包链上一级闭包中的标识符(localDeclVars和localDeclFuncs及它自身的upvalue数组)的引用
函数实例、函数引用和闭包有什么联系呢?要回答这样的问题,就要好好分别理解上面说的这个三个分别指的是什么。然后我想在回答这个问题前,通过代码来感知一些东西。其实有时候人的思维需要一个载体去依靠,也就是我们常说的实践才能出真知。
[javascript] function myFunc() { this.doFunc = function() {} } var obj = {}; //进入myFunc,取得doFunc()的一个实例 myFunc.call(obj); //套取函数实例的一个引用并赋值给func var func = obj.doFunc; //再次进入myFunc,又取得了doFunc()的一个新实例 myFunc.call(obj); //比较两次取得的函数实例,结果显示false,表明是不同的实例 print(func === obj.doFunc); //显示false //显示true,表明两个函数实例的代码文本完全一样 print(func.toString() === obj.doFunc.toString()); //显示true [/javascript]
我两次调用了myFunc,却发现对于相同的代码文本产生了不同的实例(func和obj.doFunc)。这是什么原因呢?事实上是我们每次调用myFunc函数时,Js引擎进入myFunc函数,完成赋值操作时必然要解释那段匿名函数代码文本,所以以它为蓝本产生了一个函数实例。此后我们让obj对象的doFunc成员指向这个实例。多次调用,就多次的进入myFunc做赋值,就多次的产生新的同样的代码文本的实例,所以两次obj.doFunc成员所指的函数实例是不一样的(但函数实例的静态代码文本完全相同,这里都是一个匿名空函数)。再看一个实例与引用的例子:
[javascript] function MyObject() {} MyObject.prototype.method = function() {}; var obj1 = new MyObject(); var obj2 = new MyObject(); print(obj1.method === obj2.method); //显示true [/javascript]
这里的obj1和obj2的方法其实也是对一个函数实例的引用,但怎么就相同呢?其实这样回顾我在Js对象真经里说的,prototype原型是一个对象实例,里面维护了自己的成员表,而MyObject的对象实例会有一个指针指向这个成员表,而不是复制一份。所以obj1.method和obj2.method实质都是执行同一个函数实例的两个引用。再来:
[javascript] var OutFunc = function() { //闭包1 var MyFunc = function() {}; return function() { //闭包2 return MyFunc; } }(); //注意这里的一个调用操作 var f1 = OutFunc(); var f2 = OutFunc(); print(f1 === f2); //显示true [/javascript]
这个例子其实玩了一个视觉陷阱。如果你没有注意那个函数调用的()号,就会由前面讲的知识推导出f1和f2所指的实例不是同一个(尽管它们代码文本都一样)。但这个例子我想讲的其实是关于upvalue数组的问题,不会就忘记这个数组了吧。这里其实OutFunc最终成了一个匿名函数(function() {return MyFunc}
)实例的引用,而由于闭包1内的localDeclFuncs里的数据有被引用,所以闭包1不会被销毁。然后调用这个OutFunc函数实例返回的是MyFunc函数的实例,但仅有一个,也就是多次调用都返回同一个,其原因在于匿名函数function() {return MyFunc}
实例其实是通过它运行时产生的闭包2中的upvalue中来找到MyFunc的(位于闭包1中),而MyFunc在我的那个要你们特别留意的()操作时就产生了,在闭包1的localDeclVars中记录下来。
前面的函数实例都对应的是一个闭包,这里我想拿出一个函数实例可以对应多个闭包的例子来为等下回答函数实例、函数引用和闭包有什么联系做铺垫。
[javascript] var globalFunc; function myFunc() { //闭包1 if (globalFunc) { globalFunc(); } print('do myFunc: ' + str); var str = 'test'; if (!globalFunc) { globalFunc = function() { //闭包2 print('do globalFunc: ' + str); } } return arguments.callee; } myFunc()(); /* 输出结果: do myFunc: undefined //第一次执行时显示 do globalFunc: test //第二次执行时显示 do myFunc: undefined //第二次执行时显示 */ [/javascript]
这个例子有点复杂,请读者耐着性质冷静的分析下。这里的myFunc函数实例被调用了两次,但都是同一个myFunc的实例。这是因为第一次调用myFunc实例后返回了一个它的引用(通过return arguments.callee
)。然后立即被()操作,完成第二次调用。在第一次调用中,str还没有并赋值,但已经被声明了并被初始化为了undefined,所以显示do myFunc: undefined
。而接着后面的语句,实现了赋值,这时候str的值为“test”,由于globalFunc为undefined,所以赋值为一个匿名函数实例的引用。由于globalFunc是全局变量,属于不会销毁的全局闭包。所以这时候在第一次myFunc函数实例调用完毕后,全局闭包中的globalFunc所指的匿名函数实例引用了闭包1中的str(值为test),导致闭包1不会被销毁,等第二次myFunc函数实例被调用时,产生了新的闭包3,同时也会先声明变量str,初始化为undefined。由于全局变量globalFunc有值,所以会直接调用对应的闭包1中的那个匿名函数实例,产生了闭包2,同时闭包2通过upvalue数组取得了闭包1中的str,闭包1中当时为“test”,所以输出了do globalFunc: test
。注意,这里就体现了同一个实例多次调用都要产生新的闭包,且闭包中的数据不是共享的。
这个例子总的来说反应了几个问题:
- Js中同一个实例可能拥有多个闭包;
- Js中函数实例与闭包的生存周期是分别管理的;
- Js中函数实例被调用,总是会产生一个新的闭包,但上次调用产生的闭包是否已经销毁取决于那个闭包中是否有被其他闭包引用的数据。
讲到这里,是时候揭晓谜团了。函数实例可以有多个函数引用,而只要存在函数实例的引用,该实例就不会被销毁。而闭包是函数实例被调用时产生的,但不一定随着调用结束就销毁,一个函数实例可以同时拥有多个闭包。
闭包有些什么产生的情形呢?其实细分可以分为:
- 全局闭包;
- 具名函数实例产生的闭包;
- 匿名函数实例产生的闭包;
- 通过
new Function(bodystr)
产生的函数实例的闭包; - 通过with语句所指示的对象的闭包。
关于后面3种大家先不要急,我会在后面的闭包带来的可见性问题中举例说明。
闭包中的标识符的优先级是什么样的呢?要了解优先级,就要先说说闭包中有什么标识符。完整地说,函数实例闭包内的标识符系统包括:
- this
- localDeclVars
- 函数实例的形式参数
- arguments
- localDeclFuncs
我们先看一个例子:
[javascript] function foo1(arguments) { print(typeof arguments); } foo1(1000); //显示number function arguments() { print(typeof arguments); } arguments(); //显示object function foo2(foo2) { print(foo2); } foo2('hi'); //显示hi [/javascript]
从上面我们不难分析出,形式参数优先于arguments(由foo1知),内置对象arguments优先于函数名(由arguments()知),最后一个反应了形式参数优于函数名。其实有前面两个也说明了这点形式参数优于arguments优于函数名。再看一个例子:
[javascript] function foo(str) { var str; print(str); } foo('test'); //显示test function foo2(str) { var str = 'not test'; print(str); } foo2('test'); //显示not test [/javascript]
这个例子可以得出一个结论:
- 当形式参数名与未赋值的局部变量名重复时,取形式参数;
- 当形式参数名与有值的局部变量名重复时,取局部变量值。
而this关键字,我们不能用它去做函数名,也不能作为形式参数,所以没有冲突,可以理解为它是优先级最高的。
闭包带来的可见性问题,这不是一个提问,而是一个对于闭包理解的实践。之所以这样说,是因为其实我们遇到的很多标识符可见性的问题,其实和闭包息息相关。比如内部函数可以访问外部函数中的标识符,完全是由于内部函数实例的闭包通过其upvalue数组来获得外部函数实例闭包中的数据的。可见性覆盖的实质就是在内部函数实例闭包中能找到对应的标识符,就不会去通过upvalue数组寻找上一级闭包里的标识符了。还有比如我们如果在一个函数里面不用var关键字来声明一个标识,就会隐式的在全局声明一个这样的标识。其实这就是因为在函数实例的所有闭包中找不到对应的标识,一直到了全局闭包中,也没有,所以Js引擎就在全局闭包(这个闭包有点特殊)声明了一个这样的标识来作为对于你的代码的一种容错,所以最好不要这样去用,容易污染全局闭包的标识符系统,要知道全局闭包是不会销毁的。
这里我想补充讲一个很重要的话题,就是闭包在闭包链中的位置怎么确定?
- 全局闭包没什么好说的,必然是最顶层的;
- 对于具名函数实例产生的闭包,其实由该具名函数实例的静态语法作用域决定,也就是Js引擎做语法分析时,它处于什么语法作用域,那以后产生的闭包,在闭包链中也要按这个顺序,因为函数实例被执行时也会在这个位置去执行。也就是如果它在代码文本里表现为一个函数里的函数,那将来它的闭包就被外面函数实例产生的闭包包裹;注意:对于SpiderMonkey有点不同,在with语句里面的函数闭包链隶属于with语句打开的闭包。这在后面会特别举例。
- 匿名函数实例的闭包位置由匿名函数直接量的创建位置决定,动态的加入闭包链中,而与它是否执行过无关,因为将来要产生闭包时,匿名函数直接量还是会回到创建位置执行;
- 通过
new Function(bodystr)
产生的函数实例的闭包很有意思,它不管在哪里创建,总是直接紧紧的隶属于全局闭包,也就是其upvalue数组里的数据是全局闭包中的数据; - 通过with语句所指示的对象的闭包位置由该with语句执行时的具体位置决定,动态的加入闭包链中。
我们来看一些例子,用来说明上面的结论。
[javascript] var value = 'global value'; function myFunc() { var value = 'local value'; var foo = new Function('print(value)'); foo(); } myFunc(); //显示global value var obj = {}; var events = {m1: 'clicked', m2: 'changed'}; for (e in events) { obj[e] = function() { print(events[e]); } } obj.m1(); //显示相同 obj.m2(); //显示相同 var obj = {}; var events = {m1: 'clicked', m2: 'changed'}; for (e in events) { obj[e] = new Function('print(events["' + e + '"])'); } obj.m1(); //显示clicked obj.m2(); //显示changed [/javascript]
上面一大段代码显然可以分为3部分。第一部分说明了通过new Function(bodystr)
产生的函数实例的闭包在闭包链中的位置总是直接隶属于全局闭包,而不是myFunc的闭包。第二段显示相同的原因在于调用m1、m2时,它们各自的闭包都要引用全局闭包中的e,这时e已经是events数组for迭代中的最后一个元素的索引了。第三段只用了new Function代替了原来的function,就发生了变化,那是因为Function构造器传入的都是字符串,不会将来引用全局闭包的e了。
如果我们把闭包的可见性理解为闭包的upvalue数组和闭包内的标识符系统,那一般函数实例的闭包和通过with语句所指示的对象的闭包在前者上是一致的,而在后者上处理就不一样了。原因在于:通过with语句所指示的对象的闭包只有对象成员名可访问,而没有this、函数形式参数、自动构建赋值的内置对象arguments以及localDeclFuncs,而对于该闭包中的var声明变量的效果,就具体的引擎实现会有些差异,我觉得应该避免使用这种代码的写法,所以这里就不具体讨论了。看一些与with语句有关的代码:
[javascript] var x; with(x = {x1: 'x1', x2: 'x2'}) { x.x3 = 'x3'; //通过全局变量x访问匿名对象 for (var i in x) { print(x[i]); } } function self(x) { return x.self = x; } with(self({x1: 'x1', x2: 'x2'})) { self.x3 = 'x3'; //通过匿名对象自身的成员self访问对象自身 for (var i in self) { if (i != 'self') print(self[i]); } } [/javascript]
上面的代码里注释都写得很明白了,我就不多说了,我们现在来看一个严重的问题:
[javascript] var obj = {v: 10}; var v = 1000; with(obj) { function foo() { v *= 3; } foo(); } /* with(obj) { obj.foo = function() { v *= 3; } obj.foo(); } alert(obj.v); //显示30 alert(v); //显示1000 */ alert(obj.v); //显示10 alert(v); //显示3000 [/javascript]
这段代码在ie8、safari3.2、opera9和chrome1.0.154.48中都表示为10 3000,但在Firefox3.0.7中就变为了30 1000,这个就很特别了,也许是bug问题。这里就违背了我前面说的具名函数实例闭包在闭包链中位置的一般情况,即语法分析期就决定了,视with语句于透明。其实从代码的第一印象上说,似乎就是给obj对象的属性v乘了3,而不是全局变量v。其实在除了Firefox3.0.7(其他版本的FF没有测试过),foo函数的闭包位置在语法分析期就决定了,直接隶属于全局闭包,而with语句打开的obj对象的闭包执行的位置在决定了它也直接隶属于全局闭包,所以就出现了并列的情况,那么foo函数里面的乘3自然就无法访问with闭包里的对象obj.v了,所以显示为10 3000。如果想显示为30和1000我们完全可以利用匿名函数实例闭包位置的动态特性,就是直接创建量的位置决定闭包在闭包链中的位置。所以上面我注释的代码就可以很好的显示30和1000,且Firefox也没有问题。注意匿名函数实例闭包位置与其是否执行过无关。下面代码说明这一点:
[javascript] function foo() { function foo2() { //foo2的闭包 var msg = 'hello'; m = function(varName) { //直接创建量赋值给m时,就决定了这个匿名函数实例【将来】的闭包隶属于foo2的闭包,所以可以通过upvalue数组访问msg return eval(varName); } } foo2(); var aFormatStr = 'the value is: ${msg}'; var m; var rx = /\$\{(.*?)\}/g; print(aFormatStr.replace(rx, function($0, varName) { return m(varName); })) } foo(); //显示hello [/javascript]
好了,我要讲的讲完了,总之,闭包是执行期的概念。如果大家对于闭包还有什么问题或者对于本文有何高见,都欢迎给我留言、评论或者直接email me。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
· Sdcb Chats 技术博客:数据库 ID 选型的曲折之路 - 从 Guid 到自增 ID,再到
· 语音处理 开源项目 EchoSharp
· 《HelloGitHub》第 106 期
· Spring AI + Ollama 实现 deepseek-r1 的API服务和调用
· 使用 Dify + LLM 构建精确任务处理应用