在计算机科学中,闭包(Closure)是词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。 --- 维基百科
一个公司会有自己的房子【一个函数有自己的作用域】。
公司会接收一些原料,进行一些处理,也会返回一些东西。【函数接收参数,进行处理,会有返回值】
外面的东西,是大家公有的,所以在公司内部也能用外面的东西。【函数内部能访问全局变量】
房子内的东西是公司自己的。只能在公司内使用,外面看不到也用不了。【局部变量是函数私有的,只能在函数内使用,外部无法访问】
公司里面可以开一些实验室,这个实验室是公司内部的,所以也可以使用公司内部的设备。【在函数A内部定义的函数B,可以访问函数A的局部变量】
一个公司完成其历史使命后,就没用了,会被拆迁,里面的私有的设备也会消失。【函数执行完后,会被GC回收,里面的局部变量也会消亡】
目前为止一切都看起来很正常,那么重点来了,如果公司最后给出的不是产品,而是把内部实验室丢出来成为另一个公司呢,会怎么样?
答案就是这个公司还是会被拆迁,但那些子公司需要用到的设备却不会被回收,因为子公司还需要。于是,这个子公司就是闭包,它在原来的公司倒闭前跑出来了,继续存活,还在原来公司的私有设备上贴了个标签说”这个设备我还要用,你把原公司的其他东西回收就好,这个不要回收掉”,于是拆迁大队GC就不会回收这个设备。
于是我们就实现了,在某公司倒闭后还能使用其设备。【在函数运行完后,还能通过某种方式访问它的局部变量,函数结束了,但局部变量还在,闭包的效果出现了】
也就是【闭包会使变量始终保存在内存中】或者说【闭包是由函数和与其相关的引用环境组合而成的实体】
注意事项
在一个公司内部创建十个子公司的时候,这些子公司只会在现有的设备上贴标签,也就是十个子公司会共享一个设备
像这样
for(var i = 0; i < 10; i++) { setTimeout(function() { console.log(i); }, 1000); }
结果就会输出10个10,而不是0-9。
就好比公司有个黑板i。
在黑板i写上0,然后创立一个内部的公司,让它在明天把这个黑板上写的东西发出去。
将黑板i上的0改成1,然后再创立一个内部的公司,让它在明天把这个黑板上写的东西发出去。
黑板i上的1改成2,然后再创立一个内部的公司,让它在明天把这个黑板上写的东西发出去。
……
一共创立了十个公司。
然后公司消亡了,但黑板i被留了下来,因为还有十个子公司还要用。
明天到了,十个子公司都去看黑板i,上面写了个“10”,于是他们都向外发出了 “10”。
所以怎么做呢
for(var i = 0; i < 10; i++) { (function(e) { setTimeout(function() { console.log(e); }, 1000); })(i); }
这样就不同了,每个子公司都有自己的黑板“e“(当然可以把子公司的黑板也命名为i,一样能用,就是看起来比较蛋疼)。
创立的时候就把公司的黑板i上的东西抄到自己的黑板上,等待明天发布。
于是十个子公司,都有自己的黑板,黑板上分别是他们创立的时候的黑板i上的内容,即使后面黑板i的内容变化了,但已经创立的子公司不会去管,他只管自己家的黑板e。
明天到了,十个子公司都发布了自家黑板e上的内容,也就是0-9,没人再去管原来的黑板i了。
差不多就是这样,更具体可以看详解js闭包 和 mozilla的闭包教程,这是讲得比较好的两篇。
我也是才开始学JavaScript,有什么理解错的,欢迎指出。