事件循环

2018-09-17  本文已影响0人  One_Hund

Node.js中的定时器

这里我们只介绍3个非I/O异步接口,setTimeoutsetImmediateprocess.nextTick

在node.js的事件循环中,处理各种回调是有分多个阶段的,大致如下:

循环开始

    执行setTimout、setInterval回调

    执行I/O回调

    执行setImmediate回调

循环结束
setTimeout(callback, delay[, …arg])

计划在 delay 毫秒执行 callback。返回一个可能被 clearTimeout() 用到的 timeoutObject 。

对照上面的的循环阶段不难理解,执行setTimeout的时机是在每次循环的开头,在这个阶段将会执行所有定时器时间到了或者超过的回调。

需要注意的是这里如果上一个方法执行时间过长,可能导致定时器回调实际执行的时间被延后。

function waste(time) {
    let start = Date.now();
    while(Date.now() - start < time);
}

setTimeout(() => {
    console.log('exec after 1000ms');
    waste(5000);
}, 1000);

setTimeout(() => {
    console.log('exec after 2000ms');
}, 2000);

上面的代码1000ms后会执行第一个定时器的回调,在这个回调里面又会占用CPU至少5000ms时间,第二个定时器回调执行的时候已经要等到6000ms以后了。

setImmediate(callback, [arg], [...])

计划“马上”执行 callback。返回一个 immediateObject 以供 clearImmediate() 在有需要的时候使用。

按照上面的循环阶段,我们可以看到 setImmediate 是在循环的末尾阶段被执行。它不用指定延时时间,只要前面的I/O阶段回调被执行完立马会执行 setImmediate 指定的回调方法。

相比 setTimeout(callback, 0),setImmediate 是一个更好的选择,它在nodejs的内部实现更加高效。

在 setImmediate 指定的回调中继续调用 setImmediate,其回调将会在下一轮循环的同样阶段被执行。因此可以递归调用达到每次 tick 都触发一次回调的效果:

function execEachTick() {
    setImmediate(execEachTick);
    console.log('exec each tick');
}

setImmediate(execEachTick);
process.nextTick(callback[, ...args])

计划“立即”执行 callback。不同于 setTimeout 跟 setImmediate,它没有特定的处理阶段,而是在每个阶段后面都可以执行其设定的回调方法。调用 process.nextTick 是最快速执行异步回调的方法。

它的接口命名跟功能个人觉得跟 setImmediate 是反过来的(手动滑稽),process.nextTick 才是立即执行,而 setImmediate 是下一次tick执行。

因此在process.nextTick不能递归的调用,因为递归的调用相当于无限循环:

function fn() {
    process.nextTick(fn);
}

process.nextTick(fn);
上一篇 下一篇

猜你喜欢

热点阅读