Interviewer: What is EventLoop? You: A face of deception. Just read this article
Translated from:
https://javascript.info/event-loop
Copy the code
In this article, we are going to study with two questions
- What is the concept of EventLoop
- Why EventLoop
Event loop
Browser JS and Nodejs are based on event loops, and understanding event loops is important for code optimization. In this chapter, we first introduce the theoretical details of how things work and then introduce the practical applications of that knowledge.
There is an infinite loop: the JavaScript engine waits for tasks, performs tasks, and then goes to sleep, waiting for more tasks.
The general algorithm of the engine
- When on a mission:
- Start with the earliest tasks and execute them.
- Sleep until a task appears, then switch to a task
This is the formal information you see when you browse the page. The JavaScript engine does nothing most of the time and runs only when the script/handler/event is activated.
Task sample
<script src="..." >
When an external script is loaded, the task is to execute it- When the user moves the mouse, the task is scheduled
mousemove
Event and executes the handler - When the scheduled time comes
setTimeout
, the task is to run its callback. - . , etc.
Set tasks – the engine handles them – and then wait for more tasks (which consume near zero CPU while sleeping).
Tasks may occur when the engine is busy and then be queued.
Tasks form a queue, known as a “macro task queue” (V8 terminology) :
For example, when the engine is busy executing a script, the user may move the mouse mousemove, this setTimeout may be due to task expiration, etc., and these tasks form a queue, as shown in the figure above.
Tasks in the queue are processed on a “first come, first served” basis. The engine browser uses the finished script, which will handle the Mousemove event, then the setTimeout handler, and so on.
So far, so simple, right?
Two other details:
- The engine will never render while performing a task. It doesn’t matter if the task takes a long time. Changes to the DOM are drawn only after the task is complete.
- If one task takes too long, the browser will not be able to perform other tasks, such as handling user events. So, after a while, it raises an alert like “page unresponsive,” suggesting that the task for the entire page be terminated. This happens when there are a lot of complex calculations or programming errors that cause an infinite loop.
Use case 1: SplitCPU
task
Suppose we have a task that requires CPU.
For example, syntax highlighting (used to color the code examples on this page) is CPU intensive. To highlight the code, it performs analysis, creates a lot of color elements, and then adds them to the document – spending a lot of time writing a lot of text.
While the engine is busy with syntax highlighting, it can’t do other DOM-related work, handle user events, and so on. It may even cause the browser to “hit IC” or even “hang up” for a short period of time, which is unacceptable.
By breaking up large tasks into multiple parts, we can avoid problems. Highlight the first 100 lines, then schedule setTimeout (zero latency) for the next 100 lines, and so on.
To demonstrate this approach, for simplicity rather than text highlighting, let’s have a function that evaluates from 1 to 1000000000.
If you run the code below, the engine will “hang” for a while. For obvious server-side JS, if you’re running it in a browser, try clicking another button on the page — you’ll find that no other events are processed until the count ends.
let i = 0;
let start = Date.now();
function count() {
// do a heavy job
for (let j = 0; j < 1e9; j++) {
i++;
}
alert("Done in " + (Date.now() - start) + 'ms');
}
count();
Copy the code
The browser may even display a “script is taking too long” warning.
Let’s call a split job using nested setTimeout:
let i = 0;
let start = Date.now();
function count() {
// do a piece of the heavy job (*)
do {
i++;
} while (i % 1e6! =0);
if (i == 1e9) {
alert("Done in " + (Date.now() - start) + 'ms');
} else {
setTimeout(count); // schedule the new call (**)
}
}
count();
Copy the code
The browser interface can now be used normally during the count process.
Run count once to do part of the work, then replan itself as needed:
- First run count: I =1… 1000000.
- Second run count: I =1000001.. 2000000.
- … And so on.
Now, if onclick appears as a new auxiliary task (such as an event) while the engine is busy executing Part 1, queue it up and execute it before the next part when Part 1 completes. Regular return event loops between count executions provide enough “air” for the JavaScript engine to perform additional operations in response to other user actions.
It is worth noting that both variants of setTimeout (whether or not work is assigned) are comparable in speed. There was no significant difference in overall count times.
To make them closer together, let’s make improvements.
We move the schedule to the beginning of count() :
let i = 0;
let start = Date.now();
function count() {
// move the scheduling to the beginning
if (i < 1e9 - 1e6) {
setTimeout(count); // schedule the new call
}
do {
i++;
} while (i % 1e6! =0);
if (i == 1e9) {
alert("Done in " + (Date.now() - start) + 'ms');
}
}
count();
Copy the code
Now, when we start counting () and find that we need to do more count(), we schedule work time immediately and then do it again.
If you run it, it’s easy to notice that it takes much less time.
Why is that?
This is simple: you remember that many nested setTimeout calls have a minimum delay of 4ms in the browser. Even if we set 0, it’s 4ms (or more). So, the earlier we plan — the faster we run.
Finally, we split the task that required a lot of CPU into several parts — now it doesn’t clog the user interface. And its overall execution time won’t be much longer.
Use case 2: Progress indicator
Another benefit of assigning heavy tasks to browser scripts is that we can display progress indicators.
As mentioned earlier, changes to the DOM are drawn only after the currently running task completes, no matter how long it takes.
On the one hand, this is great, because our function might create many elements, add them one by one to the document and change their styling – visitors would not see any “intermediate” unfinished state. Is it important?
This is a demo, the changes to will not be shown until the I function is complete, so we will only see the last value:
<div id="progress"></div>
<script>
function count() {
for (let i = 0; i < 1e6; i++) {
i++;
progress.innerHTML = i;
}
}
count();
</script>
Copy the code
… But we might also want to display something during a task, such as a progress bar.
If we use setTimeout to divide heavy tasks into several parts, then changes will be drawn between them.
This looks even prettier:
<div id="progress"></div>
<script>
let i = 0;
function count() {
// do a piece of the heavy job (*)
do {
i++;
progress.innerHTML = i;
} while (i % 1e3! =0);
if (i < 1e7) {
setTimeout(count);
}
}
count();
</script>
Copy the code
Now,
Use case 3: Take action after an event occurs
In an event handler, we might decide to defer some actions until the event bubbles up and is processed at all levels. SetTimeout can be implemented by wrapping code with zero delay.
In the chapter dispatching custom events, we saw an example: the custom event menu-open is a setTimeout that is dispatched in, so it occurs after the “click” event is fully processed.
menu.onclick = function() {
// ...
// create a custom event with the clicked menu item data
let customEvent = new CustomEvent("menu-open", {
bubbles: true
});
// dispatch the custom event asynchronously
setTimeout((a)= > menu.dispatchEvent(customEvent));
};
Copy the code
Macro and micro tasks
Along with macro tasks, described in this chapter, there are microtasks, which are mentioned in the chapter.
Microtasks are only from our code. They are usually made. Then/catch/finallyPromise created: handler execution become micro tasks. Microtasks are also “secretly used” await as it is another form of promised processing.
There is also a special function called queueMicrotask(func), which can queue up microtasks for execution.
After every macro task immediately, the engine performs all tasks in the MicroTask queue before running any other macro task or rendering or anything else.
For example, take a look:
setTimeout((a)= > alert("timeout"));
Promise.resolve()
.then((a)= > alert("promise"));
alert("code");
Copy the code
In what order will this be?
- Code is shown first because it is a regular synchronous call.
- Promise shows the second because it. Then passes through the microtask queue and runs after the current code.
- Timeout is displayed last because it is a macro task.
A richer picture of the event loop is shown below (from top to bottom, i.e. : scripts first, then microtasks, rendering, etc.) :
All microtasks are completed before performing any other event processing or rendering or performing any other macro tasks.
This is important because it ensures that the application environment is essentially the same between microtasks (no mouse coordinate changes, no new network data, and so on).
If we want to execute a function asynchronously (after the current code), but before rendering changes or processing new events, we can use scheduling queueMicrotask.
This is an example with a “counting progress bar”, similar to the example shown earlier, but queueMicrotask is used instead of setTimeout. You can see it in the final render. Just like synchronizing code:
<div id="progress"></div>
<script>
let i = 0;
function count() {
// do a piece of the heavy job (*)
do {
i++;
progress.innerHTML = i;
} while (i % 1e3! =0);
if (i < 1e6) {
queueMicrotask(count);
}
}
count();
</script>
Copy the code
The profile
A more detailed event loop algorithm (although still simplified compared to the specification) :
- 1 Dequeue from the macro task queue and run the earliest task (for example, script).
- 2 Execute all microtasks: – When the microtask queue is not empty: – Queue out and run the oldest microtask.
- 3 Render changes (if any).
- 4 If the macro task queue is empty, wait until the macro task appears.
- 5 Go to Step 1.
To schedule new macro tasks:
- Use zero-delay setTimeout(f).
This can be used to break down heavy computing tasks into parts so that the browser can react to user events and display progress between them.
In addition, it is used in an event handler to schedule actions after the event has been completely processed (bubbled).
Schedule new microtasks
- Use the queueMicrotask (f).
- The Promise handler also passes through the microtask queue.
There is no UI or network event handling between microtasks: they run immediately in succession.
Therefore, you might want queueMicrotask to perform functions asynchronously, but in environment state.