文章目录
一、问题:for循环setTimeout输出1-10解决方式
1-1 块级作用域var代码及输出结果
for (var i = 0; i< 10; i++){
setTimeout((i) => {
console.log(i);
}, 0)
}
1-2 输出非1-10的原因
1、 for循环是同步代码,setTimeout是异步代码。遇到这种既包含同步又包含异步的情况,JavaScript依旧按照从上到下的顺序执行同步代码,并将异步代码插入任务队列
2、 setTimeout的第二个参数则是把执行代码(console.log(i))添加到任务队列需等待的毫秒数,但等待的时间是相对主程序完毕的时间计算的,也就是说,在执行到setTimeout函数时会等待一段时间,再将当前任务插入任务队列。
3、 最后,当执行完同步代码,js引擎就会去执行任务队列中的异步代码。但JavaScript引擎开始执行任务队列中的代码时,会开始在当前的作用域中开始找变量i,但是当前作用域中并没有对变量i进行定义。
4、 这个时候就会在创造该函数的作用域中寻找i。创建该函数的作用域就是全局作用域,这个时候就找到了for循环中的变量i,这时的i是全局变量,并且值已经确定。
二、 解决方法
方法一 、最精简解决方案
for (var i = 0; i< 10; i++){
setTimeout((i) => {
console.log(i)
}, 1000,i);
}
方法二 最优解决方案,利用let形成块级作用域
for (let i = 0; i< 10; i++){
setTimeout(() => {
console.log(i)
}, 1000);
}
方法三 IIFE(立即执行函数),类似于let生成了块级作用域。
for (var i = 0; i< 10; i++){
((i)=>{
setTimeout(() => {
console.log(i)
},1000);
})(i)
}