![e61911df8481cf58d8a18d8aa465d978.png](https://i-blog.csdnimg.cn/blog_migrate/eb78f2e2e7dd4bbacc0e37a253f922c9.jpeg)
下面代码为什么会打印出6个6
let i = 0
for(i = 0; i<6; i++){
setTimeout(()=>{
console.log(i)
},0)
}
![f4faeb741feb6f255e4e69904876326f.png](https://i-blog.csdnimg.cn/blog_migrate/8931d13231b1e91568097fab82cafd4d.jpeg)
js执行机制与作用域链
首先,JavaScript是单线程环境,代码从上到下依次执行。这种执行方这也被称作是“同步执行”。(同一时间JavaScript只能执行一段代码,如果这段代码要执行很长时间,那么之后的代码只能尽情地等待它执行完才能有机会执行)。
但JavaScript中引进了异步机制。于是,所有任务可以分成两种,一种是同步任务(synchronous),另一种是异步任务(asynchronous)。同步任务指的是,在主线程上排队执行的任务,只有前一个任务执行完毕,才能执行后一个任务;异步任务指的是,不进入主线程、而进入”任务队列”(task queue)的任务,只有主线程上的任务执行完了,才通知”任务队列”,任务队列中的任务才会进入主线程执行。
在上面的代码中,for循环是同步代码,setTimeout是异步代码。遇到这种既包含同步又包含异步的情况,JavaScript依旧按照从上到下的顺序执行同步代码,并将异步代码插入任务队列。setTimeout的第二个参数则是把执行代码(console.log(i))添加到任务队列需等待的毫秒数,但等待的时间是相对主程序完毕的时间计算的,也就是说,在执行到setTimeout函数时会等待一段时间,再将当前任务插入任务队列。
最后,当执行完同步代码,js引擎就会去执行任务队列中的异步代码。这时候任务队列中就会有十个console.log(i)。我们知道,在每次循环中将setTimeout里面的代码“console.log(i)”放入任务队列时,i的值都是不一样的。但JavaScript引擎开始执行任务队列中的代码时,会开始在当前的作用域中开始找变量i,但是当前作用域中并没有对变量i进行定义。这个时候就会在创造该函数的作用域中寻找i。创建该函数的作用域就是全局作用域,这个时候就找到了for循环中的变量i,这时的i是全局变量,并且值已经确定:6。6个console.log“共享”i的值。这就是作用域链的问题。
为了解决以上两个问题,可以使用let或者闭包或者箭头函数。
打印0、1、2、3、4、5的方法
1、let方法
let i = 0
for(let i = 0; i < 6; i++){
setTimeout(()=>{
console.log(i)
},0)
}
![8fb4b14d05a8ad752c60b328c26787ef.png](https://i-blog.csdnimg.cn/blog_migrate/9bc9f75d27c89e83c9b23c47d03534ec.png)
2、闭包方法
for (i = 0; i < 6; i++) {
!function(j){//闭包
setTimeout(function(){
console.log(j);
},0)
}(i);//闭包
};
![fda2bd46a43f391a012e2a9038bdf05f.png](https://i-blog.csdnimg.cn/blog_migrate/68b5ccb930a96fff273b148b90ca02f1.png)
引用文章:
用setTimeout实现for循环中的计时器miss-me.github.io![f22a81f5611c773b767b800720925d1b.png](https://i-blog.csdnimg.cn/blog_migrate/3dccdaaac0aca24a609f52449535d3af.jpeg)
Just for myself
Amusing Ourselves to Death