Preface:This time I will mainly sort out my understanding of the Js event loop mechanism, synchronization, asynchronous tasks, macro tasks, and micro tasks. There is a high probability that there are still some deviations or errors for the time being. If so, you are welcome to correct my mistakes! 1. Reasons for the event loop and task queue:First of all, JS is single-threaded, so this design is reasonable. Imagine if the DOM is deleted on one side and added on the other side, how should the browser handle it? Quote: " Single-threaded means that tasks are serial, and the next task needs to wait for the previous task to be executed, which may result in a long wait. However, due to tasks such as ajax network requests, setTimeout time delays, and user interactions with DOM events, these tasks do not consume CPU and are a waste of resources. Therefore, asynchrony appears. By assigning tasks to corresponding asynchronous modules for processing, the efficiency of the main thread is greatly improved, and other operations can be processed in parallel. When the asynchronous processing is completed and the main thread is idle, the main thread reads the corresponding callback and performs subsequent operations to maximize the use of the CPU. At this time, the concepts of synchronous execution and asynchronous execution appear. Synchronous execution means that the main thread executes tasks in sequence and serially; asynchronous execution means that the CPU skips waiting and processes subsequent tasks first (the CPU performs tasks in parallel with network modules, timers, etc.). This creates task queues and event loops to coordinate the work between the main thread and asynchronous modules. " two, Event loop mechanism:Diagram: First, the JS execution code operation is divided Step 1: The main thread reads the JS code, which is a synchronous environment, forming the corresponding heap and execution stack; Common asynchronous operations include: ajax request, setTimeout, and similar onclik events, etc. three, Task Queue:Synchronous and asynchronous tasks enter different execution environments respectively. Synchronous tasks enter the main thread, that is, the main execution stack, and asynchronous tasks enter the task queue. First of all, as the name suggests, since it is a queue, it follows As shown in the diagram above, there are multiple task queues, and their execution order is: In the same task queue, tasks are taken away by the main thread in the order of the queue; 3.1 Types of task queues: The task queue is divided into Macro tasks mainly include: script (whole code), setTimeout, setInterval, I/O, UI interaction events, setImmediate (Node.js environment) Microtasks mainly include: Promise, MutaionObserver, process.nextTick (Node.js environment) 3.2 The difference between the two: (1) Unique, there is only one in the entire event loop; PS: Therefore, microtask queue can form a synchronous execution environment Macrotask (1) Not unique, there is a certain priority (user I/O part has a higher priority) 3.3 A more detailed event loop process
Let's use a simple example to deepen our understanding: console.log('1, time = ' + new Date().toString()) // 1. Enter the main thread, execute synchronization tasks, and output 1 setTimeout(macroCallback, 0) // 2. Join the macro task queue // 7. Start executing this timer macro task, call macroCallback, and output 4 new Promise(function (resolve, reject) { //3. Join the microtask queue console.log('2, time = ' + new Date().toString()) //4. Execute the synchronization code in this microtask and output 2 resolve() console.log('3, time = ' + new Date().toString()) //5. Output 3 }).then(microCallback) // 6. Execute the then microtask, call microCallback, and output 5 //Function definition function macroCallback() { console.log('4, time = ' + new Date().toString()) } function microCallback() { console.log('5, time = ' + new Date().toString()) } Running results: Four, Powerful asynchronous expert process.nextTick() The first time I saw this thing, it looked familiar. I thought about it and it seemed that I had used 4.1 When is process.nextTick() called?Whenever process.nextTick() is called during a given phase, all callbacks passed to process.nextTick() will be resolved before the event loop continues. In the event loop, each loop operation is called Let's borrow other people's examples to deepen our understanding of the event loop: var flag = false // 1. Variable declaration Promise.resolve().then(() => { // 2. Distribute the then task to the microtask queue of this round of loop console.log('then1') // 8. Execute the then microtask, print then1, and the flag is true at this time flag = true }) new Promise(resolve => { // 3. Execute the synchronous code in Promise console.log('promise') resolve() setTimeout(() => { // 4. Put the task in the timer into the macro task queue console.log('timeout2') // 11. Execute the timer macro task. A waiting time of 10 is specified here, so it is executed after another timer task}, 10) }).then(function () { // 5. Distribute the then task to the microtask queue of this round of loop console.log('then2') // 9. Execute the then microtask and print then2. This round of tick ends}) function f1(f) { // 1. Function declaration f() } function f2(f) { // 1. Function declaration setTimeout(f) // 7. Put `f` in `setTimeout` into the macro task queue, wait for the current round of `tick` to be executed, and then execute it in the next event loop} f1(() => console.log('f is:', flag ? 'asynchronous' : 'synchronous')) // 6. Print `f is: synchronous` f2(() => { console.log('timeout1,', 'f is:', flag ? 'asynchronous' : 'synchronous') // 10. Execute timer macro task}) console.log('This round of macro tasks has been executed') // 7. Print Running results: The callback in process.nextTick is called after the current tick is executed and before the next macro task is executed. Official example: let bar; // This method uses an asynchronous signature, but it actually calls the callback function synchronously someAsyncApiCall(callback) { callback(); } // callback function is called before `someAsyncApiCall` completessomeAsyncApiCall(() => { // bar is not assigned any value because `someAsyncApiCall` has completed console.log('bar', bar); // undefined }); bar = 1; Use let bar; function someAsyncApiCall(callback) { process.nextTick(callback); } someAsyncApiCall(() => { console.log('bar', bar); // 1 }); bar = 1; Let's look at another example with console.log('1'); // 1. Push into the main thread execution stack and output 1 setTimeout(function () { //2. Its callback function is added to the macro task queue //7. Currently the micro task queue is empty, so take out the first item in the macro task queue and execute this task console.log('2'); // Output 2 process.nextTick(function () { // 16. The last loop ends, called before the next macro task starts, output 3 console.log('3'); }) new Promise(function (resolve) { //8. Execute the synchronization task of this promise, output 4, and the state becomes resolved console.log('4'); resolve(); }).then(function () {//9. Detect the asynchronous method then and add its callback function to the microtask queue console.log('5'); // 10. Take out the first item in the microtask queue, which is the callback of this then, execute it, and output 5 }) }) process.nextTick(function () { // 11. Once the event loop ends, execute the callback of nextTick() and output 6 console.log('6'); }) new Promise(function (resolve) { //3. Execute the synchronous task in promise and output 7, the state changes to resolved console.log('7'); resolve(); }).then(function () { //4. Detect the asynchronous method then, add its callback function to the microtask queue console.log('8'); //6. The main thread is executed, take out the first item in the microtask queue, push its callback function into the execution stack, and output 8 }) setTimeout(function () { //5. Its callback function is added to the macro task queue //12. At this moment, the micro task queue is empty, and the macro task starts to execute console.log('9'); // Output 9 process.nextTick(function () { // 17. At this moment, both the microtask and macrotask queues are empty, the loop ends automatically, executes this callback, and outputs 10 console.log('10'); }) new Promise(function (resolve) { //13. Execute the synchronization task of this promise, output 11, and the state changes console.log('11'); resolve(); }).then(function () {//14. Detect the then asynchronous method and add it to the microtask queue console.log('12');//15. Take out the first item in the microtask queue, execute this then microtask, and output 12 }) }) Running results: This process is explained in detail:
Finish! When the inspiration comes to me, I quickly write it down and check it later to see if there are any problems. You are also welcome to point out my mistakes. Let's analyze a simple example: console.log('0'); setTimeout(() => { console.log('1'); new Promise(function(resolve) { console.log('2'); resolve(); }).then(()=>{ console.log('3'); }) new Promise(resolve => { console.log('4'); for(let i=0;i<9;i++){ i == 7 && resolve(); } console.log('5'); }).then(() => { console.log('6'); }) })
Code running results: This is the end of this article about the analysis of JavaScript's event loop mechanism. For more relevant JavaScript event loop mechanism content, please search for previous articles on 123WORDPRESS.COM or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future! You may also be interested in:
|
<<: In-depth exploration of whether Mysql fuzzy query is case-sensitive
>>: HTML hyperlink style (four different states) setting example
There are some tags in XHTML that have similar fu...
First download the compressed version of mysql, t...
Using the Vue language and element components, we...
I finished reading "Patterns for Sign Up &...
The Golden Rule Always follow the same set of cod...
This article shares the specific code of the vue-...
Table of contents 1. Download MySQL 2. Unzip the ...
Record some of the places where you spent time on...
Deploy nginx with docker, it's so simple Just...
Recently, the company is preparing to develop an ...
CSS writing order 1. Position attributes (positio...
1.ssh command In Linux, you can log in to another...
Tencent QQ’s homepage has been redesigned, and Web...
First install ssh in Linux, taking centos as an e...
Say it in advance We all know that Docker can ach...