1. for (var i=1; i<=5; i++) {
setTimeout( function timer() {
console.log( i );
}, i*1000 );
}
预期:分别输出数字1~5,每秒一次,每次一个。
实际:这段代码在运行时会以每秒一次的频率输出五次6。
原因:延迟函数的回调会在循环结束时才执行。当定时器运行时即使每个迭代中执行的是setTimeout(.., 0),所有的回调函数依然是在循
环结束后才会被执行,因此会每次输出一个6 出来。
2. for (var i=1; i<=5; i++) {
(function() {
setTimeout( function timer() {
console.log( i );
}, i*1000 );
})();
}
这样能行吗?(不可以)
原因:如果作用域是空的,那么仅仅将它们进行封闭是不够的。它需要包含一点实质内容才能为我们所用。它需要有自己的变量,用来在每个迭代中储存i 的值:
3. for (var i=1; i<=5; i++) {
(function() {
var j = i;
setTimeout( function timer() {
console.log( j );
}, j*1000 );
})();
}
行了!它能正常工作了!。
可以对这段代码进行一些改进:
4. for (var i=1; i<=5; i++) {
(function(j) {
setTimeout( function timer() {
console.log( j );
}, j*1000 );
})( i );
}
总结:当函数可以记住并访问所在的词法作用域,即使函数是在当前词法作用域之外执行,这时
就产生了闭包。