background
A nice afternoon. My friend shared a top interview question with me, as follows:
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('setTimeout')},0)
async1();
new Promise(function(resolve){
console.log('promise1')
resolve();
}).then(function(){
console.log('promise2')})console.log('script end')
Copy the code
This topic is mainly to investigate the understanding of synchronous tasks, asynchronous tasks: setTimeout, promise, async/await execution order. (I suggest you do it yourself first.)
At that time, because I didn’t know much about async and await, I made a lot of wrong answers :(), so I didn’t record it, and then I went to read the article to clarify my thoughts. Write it below for future reference.
Js event polling some concepts
First, you need to understand a few concepts: synchronous task, asynchronous task, task queue, MicroTask, macroTask
A synchronization task refers to a task that is queued to be executed on the main thread. The next task can be executed only after the first task is completed.
Asynchronous tasks refer to tasks that do not enter the main thread but enter the Task queue. After the synchronization tasks are completed, the tasks in the asynchronous task queue are polled to execute the tasks
A macrotask queue is an asynchronous task distributed by the host environment. The task queue is a first-in, first-out data structure in which the first event is read by the main thread.
Microtasks are tasks that are distributed by the JS engine and are always added to the end of the current task queue. In addition, during the processing of microtasks, new microtasks are added to the end of the queue and executed. Note the difference with setTimeout(fn,0) :
SetTimeOut (fn(),0) specifies that a task should be executed at the earliest available free time on the main thread, that is, as early as possible. It adds an event to the end of the “task queue”, so it does not execute until both the synchronized task and the existing events in the “task queue” are processed.
To sum up:
Task Queue, MicroTask, macroTask
- An event loop has one or more task queues.(task queue is macrotask queue)
- Each event loop has a microtask queue.
- task queue = macrotask queue ! = microtask queue
- a task may be pushed into macrotask queue,or microtask queue
- when a task is pushed into a queue(micro/macro),we mean preparing work is finished,so the task can be executed now.
So we can get the order of js execution:
Start -> Remove all tasks from the first task queue -> Remove all tasks from the next task queue -> Remove all microTask tasks again ->… And so on and so forth
Some common macro and micro tasks:
Macrotask:
- setTimeout
- setInterval
- setImmediate
- requestAnimationFrame
- I/O
- UI rendering
microtask:
- process.nextTick
- Promises
- Object.observe
- MutationObserver
Promise, Async, and Await are all asynchronous solutions
Promise is a constructor that generates a Promise instance when called. The callback function defined in the THEN function is called when the state of the Promise changes. We all know that this callback function is not executed immediately, it is a micro task that is added to the end of the current task queue and executed before the next task starts executing.
Async /await comes in pairs, and the async labelled function returns a Promise object, which can be added to the callback using the then method. Statements following await are executed synchronously. But statements under await are added to the end of the current task queue for asynchronous execution as microtasks.
Let’s look at the answer
Don’t remember the question! Continue to read, warm preparation of the topic, do not turn up 🙂
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('setTimeout')},0)
async1();
new Promise(function(resolve){
console.log('promise1')
resolve();
}).then(function(){
console.log('promise2')})console.log('script end')
Copy the code
>= Node10 version is this result: script start -> async1 start -> async2 -> promise1 -> script end -> promise2 -> async1 end -> setTimeout
<node10 version is this result: script start -> async1 start -> async2 -> promise1 -> script end -> Async1 end -> promise2 -> setTimeout
In the end, there are two answers. Why are there two results? We can see that the order of async1 end and Promise2 is different, I guess because different versions of Node execute await differently, resulting in different points of time for code under await to enter the task queue. See How to Optimize asynchronous JavaScript programming in V8? Inside get to know the await
Simple understanding is as follows:
async function f(){
await p
console.log(1);
}
// Node.js8 and the upcoming standard should parse like this
function f(){
Promise.resolve(p).then((a)= >{
console.log(1)})}// The rest of the versions should parse as follows
function f(){
new Promise(resolve= >{
resolve(p)
}).then((a)= >{
console.log(1)})}Copy the code
The above two differences are mainly as follows:
- Resolve returns the Promise object directly if the promise. resolve argument is a Promise object, and the then function executes as soon as the Promise object changes.
- The old version parses await and regenerates a Promise object. Although the promise determines that resolve will be p, the process itself is asynchronous, meaning that the new promise’s resolve process is now queued, so the promise’s THEN will not be called immediately. Instead, the current queue is called until the resolve procedure is executed, and then the then function is executed. (The following exercise shows how to execute resolve() when the promise parameter is set to promise.)
Don’t worry about the problem. There’s only one truth. According to the recent TC39 decision, await will use the same semantics directly as promise.resolve ().
Finally, we analyze the possible implementation of this topic with the latest resolution (in the Chrome environment) :
- Define functions async1 and async2. Output ‘script start’
- Add the setTimeout callback function (macro task) to the next round of task queue. This code does not perform any asynchronous operations and waits for 0s. So the callback function is immediately placed at the beginning of the next task queue.
- Perform async1. We know that the statement before the await tag and the statement after the await tag are executed synchronously in async functions. So “async1 start”, then “async2 start”.
- The following statement is paused and placed at the end of the current queue.
- The synchronization task continues.
- Output ‘Promise1’. Put the function in then at the end of the current queue.
- Then print ‘script end’, notice that the synchronization task is finished, the current task queue is not finished, and two more microtasks have been added! Queues are fifO (first-in, first-out), so ‘async1 end’ and ‘Promise2’ are output to complete the first run of the queue.
- Then execute the next task on the task list. Execute the asynchronous function in setTimeout. Output “setTimeout”.
practice
A topic on StackOverflow
let resolvePromise = new Promise(resolve= > {
let resolvedPromise = Promise.resolve()
resolve(resolvedPromise)
})
resolvePromise.then((a)= > {
console.log('resolvePromise resolved')})let resolvedPromiseThen = Promise.resolve().then(res= > {
console.log('promise1')
})
resolvedPromiseThen
.then((a)= > {
console.log('promise2')
})
.then((a)= > {
console.log('promise3')})Copy the code
Result: PromisE1 -> promise2 -> resolvePromise resolved -> promise3
This problem is really very confusing. Why ‘resolvePromise Resolved ‘on the third line? I discussed it with my roommate all night without success.
Resolve a Promise object. Resolve a Promise object. We know that if promise.resolve () takes a Promise object, the Promise object will be returned directly. But when resolve() takes a Promise object, the situation is different:
resolve(resolvedPromise)
// Equivalent to:
Promise.resolve().then((a)= > resolvedPromise.then(resolve));
Copy the code
So the first time I do this, when I get here:
- In the first then function
() => resolvedPromise.then(resolve, reject)
For microtask. Is placed at the bottom of the current task list - Then Promise1 is put at the end of the task list.
- As the resolvedPromise is an resolvedPromise, execute the THEN function, place the resole() function at the end of the current queue, and output Promise1.
- Place Promise2 at the end of the queue. Perform resole ()
- Finally, the resolvePromise becomes an Resolved Promise object. Put ‘resolvePromise Resolved’ at the end of the current task list. Output Promise2.
- Place Promise3 at the end of the current task queue. Output resolvePromise Resolved. Finally, Promise3 is output.
The end! The code below is important to explain how JS will implement these new features.
Finally, if you are wrong, please correct me.
Reference:
See JavaScript asynchronous task execution order via MicroTasks and MacroTasks
More on the Event Loop
Async /await results are inconsistent between Chrome and Node.
What’s the difference between resolve(thenable) and resolve(‘non-thenable-object’)?