一、什么是闭包
当创建一个 function 的时候,引擎内部做的,将创建这个 function 所在的作用域内的 新创建的变量保存在了这个函数内部,这就是闭包,可以认为这是一种内部表达式。
对于开发者而言,闭包是一种现象。可以访问函数外部的变量。
二、闭包产生的原因
理论上创建一个 function 的时候就会产生闭包,只不过网络上大部分两个函数嵌套的闭包的代码,类似于早期的模块化开发的代码,都是为了更加直观体现闭包这种现象的代码。
我们知道,标识符的查询是根据作用域链一层一层向上查询。当我们访问函数内部的一个标识符的时候,函数体内部没有声明它,那么会查询作用域链上一个作用域节点。
因为每一个作用域节点都对应一个变量对象,那些和当前执行的 funcition 同一级创建的变量就保存在这个变量对象中,这时我们就可以访问到函数外的变量了。
由此可以得到结论:函数可以访问哪些变量,与函数在哪里调用无关,与函数在哪里被创建有关。
三、闭包造成内存泄漏的原因
内存泄漏的原因是,我有某个变量,一直没有被回收。
对于闭包,我的函数对外部变量有引用,而我不确定什么时候执行这个函数,也就不确定什么时候访问这些闭包变量,垃圾回收机制不敢回收,那么它就一直存在闭包中。
但闭包不一定会造成内存泄漏,看你怎么写代码了。
如果你能在执行完函数后及时置空这些值,那么一切ok。
四、如何清除闭包
如果你能做到,把所有闭包对象都置空,那么理论上就能清除闭包。
通常是 x = null。
weakMap。
但是,当一个闭包的数据成为游离态的数据时,它将无法被回收。
五、浏览器memory
当 Distance 为 “-”时,为游离态数据
六、内存泄漏
该被回收的对象没有被回收,就会造成内存泄漏。
- 当我用console.log打印这个闭包变量的时候,因为它在控制台永久存在。
- 当我用eval的时候,放弃所有闭包的回收,因为不确定eval里面做了什么。
- new Function的时候,同理。
- try catch延长了作用域链,生成内部参数e不会被回收,null也不行。
- with语句操作一个没有某个属性的对象时,因为访问标识符根据作用域链查询,会绑定到window上。