These two days I saw an article introducing "How to implement timely setTimeout? 》, the article originated from an interview question: Is there any way to make setTimeout on time? For more details, please refer to Appendix [1]. After reading it, I became interested in exploring the setTimeout function, so I re-checked the relevant documents on MDN. It mentioned that [minimum delay >= 4ms], so a 0ms delay timer cannot be implemented using setTimeout. If you want to implement it, a reference link [2] is provided. The author's implementation idea is to simulate it through postMessage, bypassing the limitation of setTimeout, and thus implementing a 0ms delay timer. In short, it is to start a macro task to execute the callback. Let's take a look at how it is implemented: (function() { var timeouts = []; var messageName = "zero-timeout-message"; // Like setTimeout, but only takes a function argument. There's // no time argument (always zero) and no arguments (you have to use a closure) function setZeroTimeout(fn) { timeouts.push(fn); window.postMessage(messageName, "*"); } function handleMessage(event) { if (event.source == window && event.data == messageName) { event.stopPropagation(); if (timeouts.length > 0) { var fn = timeouts.shift(); fn(); } } } window.addEventListener("message", handleMessage, true); // Add the one thing we want added to the window object. window.setZeroTimeout = setZeroTimeout; })(); The author also provides a demo page [3]. By comparing it with setTimeout(0), the execution results in my browser are as follows:
According to the comparison results, setZeroTimeout executes hundreds of times faster than setTimeout, which is a huge improvement. What I want to discuss today is what other ways can be used to implement a 0ms delay timer in addition to the above method. First, we must make sure that our custom timer is asynchronous, and secondly, it is executed as early as possible. Speaking of asynchrony, js provides several solutions, which we can verify one by one. Before discussing various implementation methods in depth, the setTimeout comparison version provided by the agreement is as follows. The execution time of the custom implementation schemes will be compared with the setTimeout version. The code is relatively simple: (function() { let i = 0; const start = Date.now(); function test() { if(i++ < 100) { setTimeout(test); } else { console.log('setTimeout execution time:', Date.now() - start); } } setTimeout(test); })(); queueMicrotaskThe queueMicrotask API can add a microtask. It is relatively simple to use. Just pass a callback function. The specific implementation is as follows: (function() { function setZeroTimeout(fn) { queueMicrotask(fn); } let i = 0; const start = Date.now(); function test() { if(i++ < 100) { setZeroTimeout(test); } else { console.log('setZeroTimeout execution time:', Date.now() - start); } } setZeroTimeout(test); })(); By comparing with the setTimeout version, the final result is as follows:
There is a detailed introduction to this API on MDN, so I will not go into details here. According to the specification document, in most cases, it is recommended to use APIs such as requestAnimationFrame() and requestIdleCallback(), because queueMicrotask will block rendering, which is not a good practice in many cases. async/awaitAsync/await is already essential for front-end developers, and we can also use it here to achieve: (function() { async function setAsyncTimeout(fn) { Promise.resolve().then(fn); } let i = 0; const start = Date.now(); async function test() { if (i++ < 100) { await setAsyncTimeout(test); } else { console.log('setAsyncTimeout execution time:', Date.now() - start); } } setAsyncTimeout(test); })(); By comparing with the setTimeout version, the final result is as follows:
If you don't mind the trouble, you can also use Promise to implement it. In fact, they are all similar, just a little more code, which is omitted here. MessageChannelMessageChannel allows us to create a new message channel and send data through its two MessagePort properties. MessageChannel provides the concept of ports to achieve communication between ports, such as communication between workers/iframes. (function() { const channel = new MessageChannel(); function setMessageChannelTimeout(fn) { channel.port2.postMessage(null); } channel.port1.onmessage = function() { test(); }; let i = 0; const start = Date.now(); function test() { if(i++ < 100) { setMessageChannelTimeout(test); } else { console.log('setMessageChannelTimeout execution time:', Date.now() - start); } } setMessageChannelTimeout(test); })(); By comparing with the setTimeout version, the final result is as follows:
The third method takes longer to run than the previous two, because the MessageChannel generates macrotasks, while the other two methods are microtasks. Microtasks are executed first and will block the main thread, so they take longer. at last This article provides three implementation methods, all of which are based on the asynchronous solution provided by js. The implementation itself is not complicated. appendix 【1】https://mp.weixin.qq.com/s/QRIXBoKr2dMgLob3Atq9-g This concludes this article about several ways to implement 0ms delay timer in js. For more relevant js delay timer content, please search for previous articles on 123WORDPRESS.COM or continue to browse the following related articles. I hope you will support 123WORDPRESS.COM in the future! You may also be interested in:
|
<<: Detailed instructions for installing SuPHP on CentOS 7.2
>>: MySQL 8.0.16 installation and configuration graphic tutorial under macOS
This article records the installation and configu...
The vue project implements an upgraded version of...
First download the compressed version of mysql, t...
Table of contents Preface HTTP HTTP Server File S...
Table of contents 1. Basic knowledge of indexing ...
Redis Introduction Redis is completely open sourc...
Written in front A database is essentially a shar...
1. Installation environment Here is also a record...
Use the Linux chmod command to control who can ac...
Table of contents The dynamic particle effects ar...
1. Purpose Write a Flask application locally, pac...
To beautify the table, you can set different bord...
Because the data binding mechanism of Vue and oth...
This article shares the specific code of Vue intr...
HTML img produces an ugly blue border after addin...