node生命周期
在浏览器事件循环 (opens new window)中,我们了解到javascript
在浏览器中的事件循环机制,其是根据HTML5
定义的规范来实现
而在NodeJS
中,事件循环是基于libuv
实现,libuv
是一个多平台的专注于异步IO的库,如下图最右侧所示:
上图EVENT_QUEUE
给人看起来只有一个队列,但EventLoop
存在6个阶段,每个阶段都有对应的一个先进先出的回调队列
上节讲到事件循环分成了六个阶段,对应如下:
每个阶段对应一个队列,当事件循环进入某个阶段时, 将会在该阶段内执行回调,直到队列耗尽或者回调的最大数量已执行, 那么将进入下一个处理阶段
除了上述6个阶段,还存在process.nextTick
,其不属于事件循环的任何一个阶段,它属于该阶段与下阶段之间的过渡, 即本阶段执行结束, 进入下一个阶段前, 所要执行的回调,类似插队
流程图如下所示:
在Node
中,同样存在宏任务和微任务,与浏览器中的事件循环相似
微任务对应有:
宏任务对应有:
其执行顺序为:
通过上面的学习,下面开始看看题目
async function async1() {
console.log('async1 start')
await async2()
console.log('async1 end')
}
async function async2() {
console.log('async2')
}
console.log('script start')
setTimeout(function () {
console.log('setTimeout0')
}, 0)
setTimeout(function () {
console.log('setTimeout2')
}, 300)
setImmediate(() => console.log('setImmediate'));
process.nextTick(() => console.log('nextTick1'));
async1();
process.nextTick(() => console.log('nextTick2'));
new Promise(function (resolve) {
console.log('promise1')
resolve();
console.log('promise2')
}).then(function () {
console.log('promise3')
})
console.log('script end')
分析过程:
执行结果如下:
script start
async1 start
async2
promise1
promise2
script end
nextTick1
nextTick2
async1 end
promise3
setTimeout0
setImmediate
setTimeout2
1
2
3
4
5
6
7
8
9
10
11
12
13
最后有一道是关于setTimeout
与setImmediate
的输出顺序
setTimeout(() => {
console.log("setTimeout");
}, 0);
setImmediate(() => {
console.log("setImmediate");
});
1
2
3
4
5
6
7
输出情况如下:
情况一:
setTimeout
setImmediate
情况二:
setImmediate
setTimeout
1
2
3
4
5
6
7
分析下流程:
setTimeout
,虽然设置的是0毫秒触发,但实际上会被强制改成1ms,时间到了然后塞入times
阶段setImmediate
塞入check
阶段times
阶段,检查当前时间过去了1毫秒没有,如果过了1毫秒,满足setTimeout
条件,执行回调,如果没过1毫秒,跳过setImmediate
回调这里的关键在于这1ms,如果同步代码执行时间较长,进入Event Loop
的时候1毫秒已经过了,setTimeout
先执行,如果1毫秒还没到,就先执行了setImmediate