JavaScript闭包,你理解吗?
简言之,理解很多问题的关键是:闭包只有在调用的时候才进行解析。
维基百科上对闭包的定义是: Closure (also lexical closure or function closure) is a function together with a referencing environment for the non-local variablesof that function.用我拙略的语言翻译过来就是:闭包(又称“词法闭包”或“函数闭包”)是一个包含了非本地变量引用环境的函数。
闭包其实就是一个函数;如果一个函数访问了它的外部变量,那么它就是一个闭包。一个典型的例子就是全局变量的使用。所以从技术上来讲,在Javascript中,每个function都是闭包,因为它总是能访问在它外部定义的变量。
示例:
首先来看一个简单的例子:
function say667() { // Local variable that ends up within closure var num = 666; var sayAlert = function() { alert(num); } num++; return sayAlert; } var sayAlert = say667(); sayAlert()
执行结果应该弹出667而不是666,这个应该很好理解。再来看一个容易迷惑的经典例子:
function buildList(list) { var result = []; for (var i = 0; i < list.length; i++) { var item = 'item' + list[i]; result.push( function() {alert(item + ' ' + list[i])} ); } return result; } function testList() { var fnlist = buildList([1,2,3]); // using j only to help prevent confusion - could use i for (var j = 0; j < fnlist.length; j++) { fnlist[j](); } }
testList的执行结果是弹出item3 undefined窗口三次。因为这三个闭包是在同一个外部函数中定义的,item的值为最后计算的结果,但是当i跳出循环时i值为3,所以list[3]的结果为undefined.
将引用变为拷贝?
理解问题的关键是,Javascript是一门解释性的语言,一个函数内部定义的另一个函数(即闭包)只有在调用的时候才进行解析。buildList函数中定义闭包时,使用了参数"list"以及内部变量"i"的引用,而不是拷贝。因此只有当闭包执行时,也就是在testList函数中调用时,才会开始引用list和i的值并输出;而此时i的值为4,结果可想而知了!
为了达到预期的效果,我们来改造一下buildList函数,而改造的关键是在每次循环中创建变量i的拷贝,也就是将引用变为拷贝!?一种简单的方法就是使用自执行的“匿名函数”来对闭包进行包裹:
function buildList(list) { var result = []; for (var i = 0; i < list.length; i++) { (function(r){ var item = 'item' + list[r]; result.push( function() {alert(item + ' ' + list[r])} ); })(i); } return result; }
这样,在函数buildList执行的时候,匿名函数会立即执行,并把i作为参数;此时匿名函数内部的变量r相当于有了i的一个拷贝,而r的值是不会被外部的循环改变的。因此函数testList的执行结果是分别弹出“item1 1”、“item2 2”、“item3 3”。?
你理解了吗??
要小心的是,在Javascript函数参数传递的时候,只有基本类型的参数会被拷贝,对象类型的参数传递的是引用。因此,如果给匿名函数传递对象类型的参数时(没有人会这么做吧!),要小心出现意外的情况;举个变态的例子:
function buildList(list) { var result = []; var obj = {}; for (obj.i = 0; obj.i < list.length; obj.i++) { (function(r){ var item = 'item' + list[r.i]; result.push( function() {alert(item + ' ' + list[r.i])} ); })(obj); } return result; }
函数testList的执行结果是什么呢?是分别弹出“item1 undefined”、“item2 undefined”、“item3 undefined”??窗口,跟前面两种写法的结果都不一样。原因是匿名函数立即执行后,其内部变量item被正确赋值,等到testList函数运行时,闭包中引用的r.i其实就是obj对象的i变量,它的值当然是3,结果就可想而知了。
闭包,你理解了吗??