维基百科:在计算机科学中,闭包(Closure),是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即便已经离开了创造它的环境也不例外。因此,有另外一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。javascript
上面的解释不免有些抽象,为了化繁为简,本文将经过实例的方式,探究Javascript中闭包的概念及其用途。为了更好地理解闭包,我将从Javascript的变量的做用域谈起。html
有点相似于原型链(proto chain),Javascript中变量听从做用域链(scope chain)规则。
如上图所示,在Javascript中,每个函数体对应于一个做用域。当访问一个变量时,咱们会先访问当前做用域内是否有定义该变量,若是没有就会在该做用域外的做用域内寻找是否有改变量,依此类推,一直寻找到全局变量。若是全局变量中依旧没有定义该变量,就会返回undefined。java
咱们来看下下面这个例子:闭包
var milk = '外面的特仑苏' function wrapper1() { var milk = '里面的特仑苏' console.log('我要喝' + milk) //我要喝里面的特仑苏 } function wrapper2() { console.log('我要喝' + milk) //我要喝外面的特仑苏 } wrapper1() wrapper2()
在上述例子中,咱们在wrapper1函数体内定义了变量milk,所以wrapper1在寻找完当前做用域便可以获得里面的特仑苏,而在wrapper2函数体内没有定义变量milk,它会沿着做用域链去寻找全局变量,而后获得了外面的特仑苏。因此,有另外一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。app
按照做用域链的规则,咱们没法在某函数体外访问到该函数内的局部变量。可是出于某些目的咱们想在函数体外访问到函数体内的局部变量,咱们该怎么作呢?请看下面例子:函数
function wrapper1() { var milk = '里面的特仑苏' function drink() { console.log('我喝了' + milk) } return drink } var result = wrapper1() result() //我喝了里面的特仑苏
咱们在函数体内再建立一个函数,而且把这个内部函数drink做为外部函数wrapper1的返回值。咱们经过执行函数wrapper1得到了它的返回值drink,而且执行它,就成功的访问到了它的内部变量milk(里面的特仑苏)。
回想维基百科中闭包的定义,再结合上述例子:drink函数就是一个闭包,由于它引用了处于它外部的变量milk。这个被引用的外部变量milk和函数drink一同存在,即便已经离开了创造它的环境也不例外。因此,有另外一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。学习
从上述例子中不难看出,闭包的一个用途就是能够访问函数的内部变量。从而能够实现一些面向对象的功能,例如设置类的隐私变量,关于这一点能够参考《对Javascript 类、原型链、继承的理解》。
闭包的另外一个用途就是可使变量一直保存在内存之中,不被垃圾回收机制所回收。看下面这个例子:code
var change; function wrapper() { var milk = '特仑苏' function drink() { console.log('我喝了' + milk) } change = function () { milk = 'AD钙奶' } return drink } var result = wrapper() result() //我喝了特仑苏 change() result() //我喝了AD钙奶
能够看到,wrapper执行以后,milk变量一直能被访问到,缘由就是result引用了wrapper内部的drink函数,drink函数又引用了milk变量,所以它一直不会被垃圾回收机制所回收。htm
由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,因此不能滥用闭包,不然可能会形成内存泄露。解决方法是,在使用完闭包函数以后,将变量设置为undefined。好比在上例中,在使用完result以后,将result设置为null或者undefined。对象