<script> for(var i=0;i<2;i++){ setTimeout(function(){ console.log(i); },0); } </script> // 结果:2,2
js同步异步执行顺序setTimeOut面试题分析:html
打印两个2而不是0,1,跟js执行顺序有关系。面试
全部的任务分为两种,一种是同步任务,一种是异步任务。同步任务是指在主线程上排队的任务。异步任务是指不进入主线程、而进入"任务队列"(task queue)的任务,只有"任务队列"通知主线程,某个异步任务能够执行了,该任务才会进入主线程执行。异步
(1)全部同步任务都在主线程上执行,造成一个执行栈(execution context stack)。函数
(2)主线程以外,还存在一个"任务队列"(task queue)。只要异步任务有了运行结果,就在"任务队列"之中放置一个事件。oop
(3)一旦"执行栈"中的全部同步任务执行完毕,系统就会读取"任务队列",看看里面有哪些事件。那些对应的异步任务,因而结束等待状态,进入执行栈,开始执行。spa
详情看阮一峰JavaScript运行机制。连接:http://www.ruanyifeng.com/blog/2014/10/event-loop.html线程
setTimeOut函数为异步任务,for循环为同步任务,setTimeOut里的函数为回调函数。执行顺序为:同步优先,异步靠边,回调垫底。因此即便setTimeOut的时间参数是0依然会放到任务队列里,而不是主线程。主线程执行完for循环之后才执行异步任务setTimeOut。另外setTimeout()只是将事件插入了"任务队列",必须等到当前代码(执行栈)执行完,主线程才会去执行它指定的回调函数。要是当前代码耗时很长,有可能要等好久,因此并无办法保证,回调函数必定会在setTimeout()指定的时间执行。code
<script> for(var i=0;i<2;i++){ (function(){ console.log(i); }(i)) } </script>
// 结果:0,1
里边换成当即执行函数,则会打印0,1.htm