查看以下HTML / Javascript代码片段:
<html>
<head>
<script type="text/javascript">
var alerts = [];
for(var i = 0; i < 3; i++) {
alerts.push(function() { document.write(i + ', '); });
}
for (var j = 0; j < 3; j++) {
(alerts[j])();
}
for (var i = 0; i < 3; i++) {
(alerts[i])();
}
</script>
</head><body></body></html>
输出:
3, 3, 3, 0, 1, 2
这不是我所期待的 - 我期待输出0, 1, 2, 0, 1, 2,
我(错误地)假设被推入数组的匿名函数将表现为闭包,捕获创建函数时分配的i
的值 - 但它实际上显示为i
表现为全局变量。
有人能解释一下这个代码示例中i
范围内发生了什么,以及为什么匿名函数没有捕获它的值?
答案 0 :(得分:8)
范围是定义变量的函数(除了没有一个,因此它是全局的)。
您传递的匿名函数正在访问父函数(再次为全局)范围中定义的变量。
你需要一个实际的关闭。
alerts.push(
function (foo) {
return function() {
document.write(foo + ', ');
}
}(i)
);
答案 1 :(得分:6)
在Javasript中,唯一“有趣的”词法范围边界是函数体。函数中任何地方声明的任何东西(除了另一个嵌套函数之外的任何地方!)都在同一范围内。关于声明的解释方式也有一些奇怪的事情。
您的匿名函数确实充当闭包,但实例化的每个函数将共享相同的“i”。我使用的一个技巧是添加另一层功能:
for (var i = 0; i < whatever; i++) {
(function(idaho) {
whatever(function() { alert("my own private " + idaho); });
})(i);
}
在某些时候,希望所有的浏览器都支持新的“let”语句,这是一种更短,更不奇怪的方式来做基本相同的事情。