总所周知,JavaScript是单线程的,也就是说同一时间只能做一件事,那为什么JavaScript不能是多线程的呢,这跟它的用途有关,作为浏览器脚本语言,JavaScript主要用途是操作DOM,如果JavaScript同时有两个线程,同时对同一个DOM进行操作,这是浏览器该执行哪个?因此为了避免这种问题,js必须是一门单线程的语言!
所有任务可以分成两种,一种是同步任务(synchronous),另一种是异步任务(asynchronous)。同步任务指的是,在主线程上排队执行的任务,只有前一个任务执行完毕,才能执行后一个任务;异步任务指的是,不进入主线程、而进入"任务队列"(task queue)的任务,只有"任务队列"通知主线程,某个异步任务可以执行了,该任务才会进入主线程执行。
异步任务又分为两种:宏任务和微任务
常见的宏任务:setTimeout,setInterval,Ajax(网络请求),i/o(操作文件) 常见的微任务:Promise
执行栈
中的代码永远最先执行执行栈
中的代码执行完毕,会在执行宏任务队列
之前先看看微任务队列
中有没有任务,如果有会先将微任务队列
中的任务清空才会去执行宏任务队列
执行栈
和微任务队列
都执行完毕才会执行,并且在执行完每一个宏任务
之后,会去看看微任务队列
有没有新添加的任务,如果有,会先将微任务
队列中的任务清空,才会继续执行下一个宏任务
为了更好地理解Event Loop,请看下图(转引自Philip Roberts的演讲《Help, I'm stuck in an event-loop》)
console.log(111);
setTimeout(function () {
console.log(222)
for (var i = 0; i < 2; i++) {
console.log(888)
}
setTimeout(function () {
console.log(333)
}, 0)
}, 0)
for (var i = 0; i < 2; i++) {
console.log(999)
}
setTimeout(() => {
console.log(444);
}, 1000);
console.log(555)
一张图表示执行过程:
执行步骤:
console.log('script start');
setTimeout(function() {
console.log('timeout1');
}, 10);
new Promise(resolve => {
console.log('promise1');
resolve();
setTimeout(() => console.log('timeout2'), 10);
}).then(function() {
console.log('then1')
})
console.log('script end');
执行结果:
script start promise1 script end then1 timeout1 timeout2
new Promise(resolve => {
resolve(1);
Promise.resolve().then(() => console.log(2));
console.log(4)
}).then(t => console.log(t));
console.log(3);
分析:
new Promise(resolve => {
resolve(1);
Promise.resolve().then(() => {
// t2
console.log(2)
});
console.log(4)
}).then(t => {
// t1
console.log(t)
});
console.log(3);
执行流程:
Promise
实例,构造函数首先执行,所以首先输出了 4。此时 microtask 的任务有 t2
和 t1
t2
和 t1
,分别输出 2 和 1为什么会t2先执行,理由如下:
Promise.resolve()
方法允许调用时不带参数,直接返回一个resolved
状态的 Promise 对象。需要注意的是,立即resolve()
的 Promise 对象,是在本轮“事件循环”(event loop)的结束时执行,而不是在下一轮“事件循环”的开始时。