According to this answer :
"And the important thing is taking into account that then/catch callbacks in a promise are microtasks and will be executed after nextTick tasks"
Nodejs docs mentions that nextTickQueue will be processed after the current operation is pleted, regardless of the current phase of the event loop. Does that mean resolved promises callbacks does execute in neither poll phase not pending callbacks phase?
Also, this video mentions that there is a queueMicroTask queue which is processed after the current phase of the event loop. So what I understand now is :
Resolved promises callbacks will be added to queueMicroTask which will be processed after the nextTickQueue (Callbacks passed to process.nextTick) which will be processed after whatever the current phase of the event loop is.
Is that correct? If it's not please correct me, I'm really confused.
According to this answer :
"And the important thing is taking into account that then/catch callbacks in a promise are microtasks and will be executed after nextTick tasks"
Nodejs docs mentions that nextTickQueue will be processed after the current operation is pleted, regardless of the current phase of the event loop. Does that mean resolved promises callbacks does execute in neither poll phase not pending callbacks phase?
Also, this video mentions that there is a queueMicroTask queue which is processed after the current phase of the event loop. So what I understand now is :
Resolved promises callbacks will be added to queueMicroTask which will be processed after the nextTickQueue (Callbacks passed to process.nextTick) which will be processed after whatever the current phase of the event loop is.
Is that correct? If it's not please correct me, I'm really confused.
Share Improve this question edited Aug 1, 2021 at 18:20 Andy 63.6k13 gold badges71 silver badges98 bronze badges asked Aug 1, 2021 at 18:13 rayanJrayanJ 712 silver badges8 bronze badges2 Answers
Reset to default 13There's an important semantic with microtasks and nextTicks which depends on the Node version.
Before Node v11, nextTick queue was executed between each phase of the event loop (timers, I/O, immediates, close handlers are the four phases). Therefore, before Node v11, promise callbacks were also executed between each phase in the event loop. (I've written about this in detail here: https://blog.insiderattack/promises-next-ticks-and-immediates-nodejs-event-loop-part-3-9226cbe7a6aa)
However, starting from Node v11, event loop jumps to microtask
queue whenever a microtask
is added to the microtask queue as part of the execution of the program. You can experiment this with the following snippet. The same applies to nextTick queue too. You can read more here: https://blog.insiderattack/new-changes-to-timers-and-microtasks-from-node-v11-0-0-and-above-68d112743eb3
setImmediate(() => console.log('timeout1'));
setImmediate(() => {
console.log('timeout2')
Promise.resolve().then(() => console.log('promise'))
});
setImmediate(() => console.log('timeout3'));
setImmediate(() => console.log('timeout4'));
The output of the above code changes depending on the Node.js version as follows:
$ node -v
v10.19.0
$ node test.js
timeout1
timeout2
timeout3
timeout4
promise
$ nvm use 11
Now using node v11.15.0 (npm v6.7.0)
$ node test.js
timeout1
timeout2
promise
timeout3
timeout4
Therefore it's important to know that: nextTicks
and microtasks
have an even higher priority in Node versions >=11, because they get the chance to be processed within the current phase of the event loop. But in earlier Node versions, nextTicks
and microtasks
are executed at the end of each phase of the loop.
On a side note, it's important to know that
microtasks
queue is a part ofv8
engine and not maintained in Node.js runtime. However, Node.js event loop instructsv8
to run allmicrotasks
, once Node.js finishes with thenextTick
queue. Therefore, promise callbacks are executed afternextTick
queue.
Nodejs docs mentions that nextTickQueue will be processed after the current operation is pleted, regardless of the current phase of the event loop. Does that mean resolved promises callbacks does execute in neither poll phase not pending callbacks phase?
It's sort of an ambiguous message for sure. They can run at either phase but they are not exclusive to a particular phase.
Resolved promises callbacks will be added to queueMicroTask which will be processed after the nextTickQueue (Callbacks passed to process.nextTick) which will be processed after whatever the current phase of the event loop is.
Not at the end of each phase, it could happen at any point during any phase (as I understand it).
Say you have two timers which have expired and you are at the timer expired phase of the event loop. Meaning there's two callbacks to deal with during this phase. Lets assume the first timer calls then
on a promise which has already resolved (it can get to it right away). If this was fired at the end of the phase, then the second timer callback would run before since it's not the end of the phase yet.
What really happens here though is that the first timer callback runs and adds the callback to the microtask queue which gets consumed right AFTER the callback finishes and BEFORE continuing down the current event loop phases.
An example you can run:
setTimeout(() => {
console.log('first');
let p = Promise.resolve();
p.then(() => console.log('second'));
p.then(() => console.log('third'));
}, 1)
setTimeout(() => {
console.log('fourth')
}, 1);