在for()循环中声明的Javascript变量的范围是什么?

时间:2010-04-28 21:57:15

标签: javascript scope puzzle anonymous-function

查看以下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范围内发生了什么,以及为什么匿名函数没有捕获它的值?

2 个答案:

答案 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”语句,这是一种更短,更不奇怪的方式来做基本相同的事情。