简单的对闭包做一个理解:
- 闭包产生是:函数内嵌套函数;
- 闭包函数在外面被调用,访问;
- 闭包能使外部函数访问内部函数的变量;
- 闭包可以保持对定义时的作用域的引用,使之不被销毁,就长生不老。
在看闭包之前我们还需要理解作用域。
作用域主要分为两种:全局变量和局部变量。
全局变量&局部变量
局部变量:定义在函数内部,只能在函数中使用的变量,作用范围是从函数开始到结尾,即在{}里。
这里定义的a
就是局部变量。
1 | function myTest1(){ |
全局变量:常常定义在函数外部,拥有全局作用域,即在 JavaScript 代码的任何地方都可以访问
这里定义的b
就是局部变量。
1 | var b = 3;//全局变量 |
从这里我们就可以知道,函数内部可以调用全局变量,但是函数外部无法读取函数内的局部变量
函数内部声明变量的时候,必须使用var命令去声明一个变量。如果不用的话,就会声明了一个全局变量。
1 | function myTest3(){ |
在这里我们c
和d
都是在函数内部声明的,但是因为c
使用了var
,而d
没有使用var
声明,所以在外部能够使用d
;也就是所谓的声明了一个全局变量
闭包
在理解了全局变量和局部变量之后,我们先写一个简单的闭包实例给大家看看:
1 | function myTest4(){ |
在这里我们是不是有点明白了呢:如果我们想用一个变量或者得到一个变量,但是我们又不想全局声明定义这个变量,到这的时候我们就可以通过闭包来调用函数内部的变量。
但是我们也应该知道闭包产生的值是一直保存在内存中的。这里通过一个例子给大家看看:
1 | function myTest5(){ |
这个例子就是说调用一次,然后i
的值就变了一次并且会一直被保存,然后下次调用的时候就是上次保存的值了。
一些闭包实例:
/* 例子1 */
1 | function outerFn(){ |
/* 例子2 */
1 | function m1(){ |
/* 例子3 */
1 | var lis = document.getElementsByTagName("li"); |
回收
由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE浏览器中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。
一个简单的回收实例:
1 | function myTest6(){ |
总结
闭包的缺点就是常驻内存,闭包会使变量始终保存在内存中,如果使用不当会增大内存使用量,造成内存泄露。
正常情况下外部函数是不能访问内部函数的变量的,但是可以利用闭包来实现对函数内部变量的访问。
一般函数执行完毕后,局部活动对象就被销毁,内存中仅仅保存全局作用域。但闭包的情况不同!