循环中的闭包问题
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了循环中的闭包问题相关的知识,希望对你有一定的参考价值。
循环中使用闭包,假设我们需要在每次循环中调用循环序号for(var i = 0; i < 10; i++) {
setTimeout(function() {
console.log(i);
}, 1000);
}
上面的代码不会输出数字 0 到 9,而是会输出数字 10 十次。
当 console.log 被调用的时候, 匿名函数保持对外部变量 i 的引用,此时 for循环已经结束, i 的值被修改成了 10.
为了得到想要的结果,需要在每次循环中创建变量 i 的 拷贝。
避免引用错误
为了正确的获得循环序号,最好使用 匿名包裹器。
for(var i = 0; i < 10; i++) {
(function(e) {
setTimeout(function() {
console.log(e);
}, 1000);
})(i);
}
外部的匿名函数会立即执行,并把 i 作为它的参数,此时函数内 e 变量就拥有了 i 的一个拷贝。
当传递给 setTimeout 的匿名函数执行时,它就拥有了对 e 的引用,而这个值是 不会被循环改变的。
有另一个方法完成同样的工作;那就是从匿名包装器中返回一个函数。这和上面的代码效果一样。
for(var i = 0; i < 10; i++) {
setTimeout((function(e) {
return function() {
console.log(e);
}
})(i), 1000)
}
以上是关于循环中的闭包问题的主要内容,如果未能解决你的问题,请参考以下文章
Swift之常见闭包与defer关键字的使用分析和闭包中的循环引用
深入理解JavaScript的闭包特性 如何给循环中的对象添加事件