闭包:是指有权访问另一个函数作用域中的变量的函数。
创建闭包的常见方式,是在一个函数内部创建另外一个函数:
如上,看Fn函数return出来的内部函数(匿名函数)中的前两行代码,这个两行代码访问了外部函数中的变量propertyname,即使该函数被返回,而且在其他地方调用了,但是他任然可以访问propertyname这个变量。(一句话理解闭包:JavaScript中的函数运行在它们被定义的作用域里,而不是它们被执行的作用域里)。这是因为在内部函数的作用域链中包含了外部函数Fn的作用域。
了解闭包之前,让我们了解下如何创建作用域链以及作用域链的作用是什么?
首先,当某个函数被调用的时候,会创建一个执行环境以及相应的作用域链,然后使用arguments和其他命名参数的值初始化函数的活动对象。在函数执行的过程中,就需要在作用域链中查找变量。如下举个栗子:
如上代码,首先是定义了一个Fn函数,然后在全局中调用了它,执行Fn()的时候,会创建一个包含arguments,val1,val2的活动对象,全局执行环境的变量对象包含有res,Fn。那么全局执行环境的作用域则是处于Fn执行环境的作用域链的第二位。
在后台的每个执行环境都有一个表示变量的对象--变量对象。全局环境中的变量对象是始终存在的,然后像Fn函数这样局部环境的变量对象,只有在函数执行的过程中存在。在创建Fn函数的时候会预先创建一个包含全局变量对象的作用域链。这个作用域链保存在内部的[[Scope]]属性中。在调用Fn函数的时候,会创建一个执行环境,然后通过复制函数的[[Scope]]属性中的对象构建起执行环境的作用域链,此后,又有一个活动对象(在此作为变量对象使用)被创建并推入执行环境作用域链的前端。
以上有三关键词:执行环境,变量对象,活动对象。注意理清楚他们之间的关系。
一般情况下,当函数执行完毕后,局部活动对象就会被销毁,内存中只会保存全局执行环境的变量对象。但是闭包却不是这样。举个栗子:
以上函数Wrapper中的匿名函数会将外部函数的活动对象添加到他的作用域链中,即返回出的匿名函数的作用域链被初始化为包含Wrapper函数的活动对象和全局的活动对象。这样Wrapper()函数执行完成后,其活动对象不会被销毁,因为匿名函数的作用域链任然存在,而且引用着外部函数的活动对象,直到匿名函数被销毁后,外面的函数的活动对象才会全部销毁。
--------以上根据javascript高级程序设计总结。我弄明白了,你呢?