Javascript函数作用域[Javascript Essentials]
Javascript function scope [Javascript Essentials]
我正在看Travis Tidwell的Javascript Essentials,他在那里解释了这段代码:
(function() {
var messages = ['hello', 'there'];
for (var i in messages) {
setTimeout(function() {
console.log(messages[i]);
}, 10);
};
})();
它在控制台上重复了两次'there',但我仍然不明白为什么。谁能一步一步地跟我讲讲这段javascript ?
每次代码在循环中运行时,它都会设置一个事件处理程序,以便在10ms后,它记录messages[i]
的值。
在这些超时时间的10ms过去之前,i
的值已经被(通过for
循环)更改为1
(因为这是数组中的最后一个属性名)。
第一次超时输出messages[1]
,第二次超时输出messages[1]
。
- 数组创建并存储在
messages
中 -
i
设置为0
并设置超时 -
i
设置为1
并设置超时 - 第一个超时函数运行,
i
仍然是1
- 第二个超时函数运行,
i
仍然是1
JavaScript有函数作用域,而不像其他语言那样有块作用域。因此,实际上只存在一个i
变量。当调用setTimeout
中的代码时,i
已经被设置为数组的最后一个索引。
let
声明块作用域变量。Mozilla Reference: let
在此之前,绕过此行为的一种方法是为需要独立于其他变量的变量创建一个新函数:
(function() {
var messages = ['hello', 'there'];
for (var i in messages) {
(function(currentIndex) {
setTimeout(function() {
logToOutput(messages[currentIndex]);
}, 10);
})(i);
};
})();
function logToOutput(msg) {
document.getElementById("output").innerHTML += msg + "<br>";
}
<div id="output"></div>
相关文章:
- 为什么不't Javascript对我的输入值进行了一些重新检查
- 创建一个类似链接的按钮,并通过Javascript函数打开一个新的弹出窗口
- 将函数的上下文应用于javascript变量
- 使用php或javascript从facebook相册URL中删除多余的部分
- 正在添加'X'按钮,在文本字段旁边使用javascript
- 如何在JavaScript中将字符串转换为函数引用
- 模糊事件的Javascript测试
- Javascript更改图标
- 如何将HTML id分配给元素,以及如何将JavaScript应用于元素
- 如何使用WCF服务和javascript表单post上传.doc文件
- javascript结合了数组和字典
- 这是什么 ==- javascript 运算符
- 从javascript创建一个列表
- 无法在通过jQuery的ajax加载的页面中执行javascript
- Javascript:selenium Web驱动程序isDisplayed()不工作
- 如何通过ajax刷新JSF填充的javascript变量
- 如何在Javascript中将JSon对象转换为数组
- Javascript生成的表单未提交
- 使用javascript将动态表从一个html页面打印到另一个html页
- 通过javascript重定向html传递php变量