Bootstrap

重生!带你深入理解JavaScript的闭包

对于那些有一点 JavaScript 使用经验但从未真正理解闭包概念的人来说,理解闭包可以看作是某种意义上的重生,但是需要付出非常多的努力和牺牲才能理解这个概念。

                                                                                               ——《你不知道的JavaScript》

在JavaScript中的”神兽“,很多小伙伴会觉得闭包这玩意太恶心了,怎么着都理解不了...其实刚接触JavaScript的时候我也是这样。

但是!!!闭包真的非常重要!非常重要!非常重要!重要的事情说三遍!!!

接下来,我会带着大家真正意义上的理解闭包。

目录

一 、闭包概念描述

二 、闭包前置知识点

1、作用域

2、作用域链

2.1 概念

2.2 使用规则

2.3 创建规则

3、词法作用域

三、解释闭包

四、再次解释闭包

五、闭包的应用

1、私有化全局变量

2、外部访问函数内部变量

3、构建私有作用域

4、模块输出

六、闭包的副作用

1、在函数中使用定时器,形成闭包,导致内存泄露

2、闭包返回被外部变量引用,导致内存泄露


一 、闭包概念描述

《JavaScript权威指南》这样描述:

函数对象可以通过作用域链相互关联起来,函数体内部的变量都可以保存在函数作用域内,这就是叫闭包。

《你不知道的JavaScript》这样描述:

闭包是基于词法作用域书写代码时所产生的自然结果。

当函数可以记住并访问所在的词法作用域时,就产生了闭包,即使函数是在当前词法作用域之外执行。

从以上的描述。要真正理解闭包概念,要先深刻理解以下几个知识点,可以称为闭包前置知识点

二 、闭包前置知识点

1、作用域

《你不知道的JavaScript》这样描述:

作用域可以理解为一套规则,来定义变量存储在哪里,使用的时候怎么找到他们。

作用域是负责收集并维护由变量组成的一系列查询,并实施一套非常严格的规则,确定当前执行的代码对变量的访问权限。

而我是这么理解的:作用域就是一个独立的对象,里面存储了变量,在对象中定义了一系列的规则,来限制外部访问里面的变量,来区分变量让不同作用域下同名变量不会有冲突。

如下图所示,红框区域就是一个作用域

2、作用域链

2.1 概念

作用域链可以理解为一个全局对象。在不包含嵌套的函数体,作用域链上有两个对象,第一个定义函数参数和局部变量的对象,第二个是全局对象。在一个嵌套的函数体内,作用域链上至少有三个对象。

举个栗子,如图所示这是不包含嵌套的函数体的作用域链

举个栗子,如图所示这是包含嵌套的函数体的作用域链

2.2 使用规则

当一个块或函数嵌套在另一个块或函数中时,就发生了作用域的嵌套。因此,在当前作用域中无法找到某个变量时,就会在外层嵌套的作用域中继续查找,直到找到该变量,或抵达最外层的作用域(也就是全局作用域)为止。

举个栗子,如图所示

比如要在作用域3中查找变量a的值,但是发现作用域3中没有变量a,就去作用域2中找,发现了变量a就停止了查找。

注意:在查找过程中不会跑去作用域4中查找。因为作用域4不是作用域3的外层嵌套作用域

2.3 创建规则

理解作用域链的创建规则对理解闭包是非常重要的

首先我们定义一个函数的时候,开始就创建并保存了一条作用域链,里面包含一个全局作用域对象,当函数被调用时,会创建一个新对象(作用域)来存储它的变量,并将这个对象添加到开始创建的作用域链上,同时创建一条新的表示调用函数的作用域的“链”。

仔细琢磨一下下面代码,就可以理解。

func
;