什么是闭包
闭包是一种抽象概念,理解起来可能比较难困难,再说闭包之前我们先了解下JavaScript的垃圾回收机制。
随着程序的运行,他们会因各种事情而占用计算机的内存。如果程序运行了却从不释放不在需要的内存,电脑最终会崩溃。在一些语言中。像C,内存管理是由程序员处理的。但在JavaScript中实现了自动回收机制。
当函数执行完毕时,管理内存的当地方法会将函数中所有创建了的东西从内存中移除。
var prison = function(){
var prisoner = 'Joke';
}
prison();
一旦prison执行完毕,我们就不需要访问prisoner变量了。
var prison = (function(){
var prisoner = 'Joke';
return {prisoner:prisoner}
})()
console.log(prisoner:prisoner)
这和上述一样prisoner:prisoner并不会访问函数中的prisoner变量,但下面这个就不一样了:
var prison = (function(){
var prisoner = 'Joke';
return {
prisoner:function(){
return prisoner;
}
}
})()
console.log(prisoner:prisoner())
现在每次执行prisoner:prisoner(),都会访问prison函数中的prisoner变量,这样垃圾回收机制就不能移除 prisoner变量。现在我们可以回答什么是闭包了,闭包就是阻止垃圾回收器将变量从内存中移除的方法,使得在创建变量的执行环境的外面能够访问到改变量。
让我们再多看几个闭包的事例:
var makePrison = function(priosner){
return function(){
return priosner;
}
}
var jokePrison = makePrison('Joke');
var mikePrison = makePrison('Mike');
闭包的原理
我们知道了什么是闭包,但是他是怎么工作的呢?答案就在执行环境对象中。我们看一下上面的事例:
当调用makePrison时,为了这次特定的调用创建了一个执行环境对象,将传入的值赋予给priosner(执行环境对象是JavaScript引擎的一部分,在JavaScript中不能直接访问)。
我们调用了两次makePrison将结果保存在 jokePrison 和mikePrison中。因为makePrison返回的是一个函数,当我们把结果赋值给jokePrison时,这个特定的执行环境对象的引用计数置为1,因为引用计数大于0,垃圾回收器不会对这个对象进行回收。
当我们再次调用makePrison,并将它的返回值赋值给mikePrison时,创建了一个新的执行环境对象,并且它的引用计数也是1。尽管它们是通过同一个函数创建的,但它们是不同的。
如果再次调用jokePriso,它会使用 在调用makePrison 时创建的并且保存给jokePriso的执行环境对象上设置的值。
tips
每次调用函数时都会创建一个唯一的执行环境对象,函数执行完后,执行对象就会被丢弃,除非调用者引用了它。当然,如果函数返回的数字,就不能引用函数的执行环境。