返回顶部
首页 > 资讯 > 前端开发 > JavaScript >js实现0ms延时定时器的几种方式
  • 264
分享到

js实现0ms延时定时器的几种方式

2024-04-02 19:04:59 264人浏览 泡泡鱼
摘要

目录queueMicrotask async/await MessageChannel 最后 附录 这两天看到一篇介绍《如何实现准时的 setTimeout?》的文章,文章起源于一道

这两天看到一篇介绍《如何实现准时的 setTimeout?》的文章,文章起源于一道面试题:有什么办法让setTimeout准时呀?具体文章内容可查看附录【1】,看完之后,引起了我对setTimeout这个函数的探究兴趣,因此在MDN上重新查阅了相关文档,其中提到【最小延时 >=4ms】的一点,因此使用setTimeout不能实现0ms延时的定时器,如果要实现的话,提供了一个参考链接【2】,作者的实现思路是通过postMessage来模拟,绕过setTimeout的限制,从而实现0ms延时的定时器,说简单来讲就是起了一个宏任务去执行回调,先具体看下是怎么实现的:


(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;
})();

作者还提供了一个demo页面【3】,通过于setTimeout(0)进行对比,在我浏览器的执行结果如下:

100 iterations of setZeroTimeout took 15 milliseconds.
100 iterations of setTimeout(0) took 488 milliseconds.

根据结果对比来看,setZeroTimeout执行比setTimeout快了上百倍,这是一个巨大的提升。今天想讨论的是除了上述这种方式,还可以通过哪些方式来实现一个0ms延时的定时器呢,首先,我们要确定一下我们自定义的定时器是异步的,其次是尽可能早的被执行。说起异步,js提供了好几种解决方案,我们可以逐一去验证。

在深入讨论各种实现方式之前,约定提供的setTimeout对比版本如下,后面自定义实现的方案都将和setTimeout版本的执行时间进行对比,代码比较简单:


(function() {
 let i = 0;
 const start = Date.now();
 function test() {
  if(i++ < 100) {
   setTimeout(test);
  } else {
   console.log('setTimeout执行时间:', Date.now() - start);
  }
 }
 setTimeout(test);
})();

queueMicrotask

queueMicrotask这个api可以添加一个微任务,使用比较简单,直接传递一个回调函数即可,具体实现如下:


(function() {
 function setZeroTimeout(fn) {
  queueMicrotask(fn);
 }
 let i = 0;
 const start = Date.now();
 function test() {
  if(i++ < 100) {
   setZeroTimeout(test);
  } else {
   console.log('setZeroTimeout执行时间:', Date.now() - start);
  }
 }
 setZeroTimeout(test);
})();

通过和setTimeout版本进行对比,最终结果如下:

setZeroTimeout执行时间: 2
setTimeout执行时间: 490

关于这个API的介绍在MDN上有详细的说明,就不展开介绍了,这里多说一点,根据规范文档的说明,大多数情况下,推荐使用requestAnimationFrame()和requestIdleCallback()等api,因为queueMicrotask会阻塞渲染,在很多时候都不是一种好的实践。

async/await

async/await对于前端开发人员来说已经是必不可少的了,这里我们也可以用来实现:


(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执行时间:', Date.now() - start);
  }
 }
 setAsyncTimeout(test);
})();

通过和setTimeout版本进行对比,最终结果如下:

setAsyncTimeout执行时间: 2
setTimeout执行时间: 490

如果不嫌麻烦,还可以通过Promise来实现,其实都是大同小异,无非多些点代码,这里就省略了。

MessageChannel

MessageChannel允许我们创建一个新的消息通道,并通过它的两个MessagePort属性发送数据,MessageChannel提供端口的概念,实现端口之间的通信,比如worker/iframe之间的通信。


(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执行时间:', Date.now() - start);
  }
 }
 setMessageChannelTimeout(test);
})();

通过和setTimeout版本进行对比,最终结果如下:

setMessageChannelTimeout执行时间: 4
setTimeout执行时间: 490

第三种方式运行时间比前面两种更长些,因为通过MessageChannel产生的是宏任务,其他两种是微任务,微任务执行靠前,且会阻塞主线程,因此时间会长一点。

最后

本文提供了三种实现方式,都是围绕js提供异步解决方案来实现的,实现本身并不复杂。

附录

​​【1】https://mp.weixin.qq.com/s/QRIXBoKr2dMgLob3Atq9-g
【2】Https://dbaron.org/log/20100309-faster-timeouts
【3】https://dbaron.org/mozilla/zero-timeout

到此这篇关于js实现0ms延时定时器的几种方式的文章就介绍到这了,更多相关js 延时定时器 内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: js实现0ms延时定时器的几种方式

本文链接: https://lsjlt.com/news/131224.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • js实现0ms延时定时器的几种方式
    目录queueMicrotask async/await MessageChannel 最后 附录 这两天看到一篇介绍《如何实现准时的 setTimeout?》的文章,文章起源于一道...
    99+
    2024-04-02
  • js如何实现0ms延时定时器
    小编给大家分享一下js如何实现0ms延时定时器,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!示例如下(function() { var&nbs...
    99+
    2023-06-20
  • Java定时器实现的方式有哪几种
    Java中实现定时器的方式有以下几种:1. 使用Timer类:Timer类是Java提供的一个定时任务调度器。可以使用Timer类来...
    99+
    2023-10-12
    Java
  • C#代码延时的几种实现
    目录Task.Delay();异步实现线程实现:计时器方式实现:Task.Delay();异步实现 using System; using System.Threading.Ta...
    99+
    2024-04-02
  • 延时任务的四种实现方式
    什么是延迟任务? 顾明思议,我们把需要延迟执行的任务叫做延迟任务。 延迟任务的使用场景有以下这些: 红包 24 小时未被查收,需要延迟执退还业务; 每个月账单日,需要给用户发送当月的对账单; 订单下单之后 30 分钟后,用户如果没...
    99+
    2023-10-04
    java Powered by 金山文档
  • Java 定时器的多种实现方式
    目录一、前言 (1)Timer (2)DelayedQueue 延迟队列 (3)ScheduledThreadPoolExecutor (4)ScheduledThreadPoolE...
    99+
    2024-04-02
  • Springboot+rabbitmq实现延时队列的两种方式
    目录什么是延时队列,延时队列应用于什么场景利用TTL DLX实现延时队列的方式TTL DLX是什么Springboot集成rabbitmq实现第一种方式利用Rabbitmq的插件x-...
    99+
    2024-04-02
  • 【Android】之【延时执行的几种方法】
    在Android日常开发中我们经常会有延时执行某个操作的需求,例如我们启动应用的时候,开始呈现一个引导页面,过了两三秒后,自动跳转到主界面;又如为了控制时序,需要延时执行等等。 下面整理下实现延时执行...
    99+
    2023-08-31
    android
  • JS格式化时间的几种方法总结
    方法一 方法 //格式化时间 function format(dat){ //获取年月日,时间 var year = dat.getFullYear(); v...
    99+
    2024-04-02
  • iOS中几种定时器的实现小结
    在软件开发过程中,我们常常需要在某个时间后执行某个方法,或者是按照某个周期一直执行某个方法。在这个时候,我们就需要用到定时器。 然而,在iOS中有很多方法完成以上的任务,到底有多少种...
    99+
    2022-05-30
    iOS 定时器
  • redis实现延时队列的两种方式(小结)
    背景 项目中的流程监控,有几种节点,需要监控每一个节点是否超时。按传统的做法,肯定是通过定时任务,去扫描然后判断,但是定时任务有缺点:1,数据量大会慢;2,时间不好控制,太短,怕一...
    99+
    2024-04-02
  • MySQL时间盲注的五种延时方法实现
    MySQL时间盲注五种延时方法 (PWNHUB 非预期解) 延时注入函数 五种:sleep(),benchmark(t,exp),笛卡尔积,GET_LOCK() RLIKE正则 sl...
    99+
    2024-04-02
  • Android 开发延时调用的几种方法
      我们在安卓开发中,有时候会遇到需要一个程序推迟一段时间执行,也就是延时执行,比如说App首页显示定格3秒,然后自动跳到登录页的情况,这个怎么做呢?本文参...
    99+
    2022-06-06
    调用 方法 Android
  • python 实现定时任务的四种方式
    目录用Python实现定时任务 用Python实现定时任务的四种方法 利用while True: + sleep()实现定时任务 利用threading.Timer定时器实现定时任务...
    99+
    2024-04-02
  • Python3-定时任务四种实现方式
    老猫最近做一个小程序开发任务,主要负责后台部分开发;根据项目需求老猫需要实现三个定时任务: 1>定时更新微信token,需要2小时更新一次;2>商品定时上线;3>定时检测后台服务是否存活; 老猫使用Python去实现这...
    99+
    2023-01-31
    四种 方式
  • Android定时器实现的几种方式整理及removeCallbacks失效问题解决
    实现定时器有很多种方式,在这里我简单的介绍几种方式 (1)使用Handler + Runnable的方式 代码如下: Handler handler = new Handle...
    99+
    2022-06-06
    Android
  • javascript的定时器有哪几种
    本篇内容介绍了“javascript的定时器有哪几种”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成! ...
    99+
    2024-04-02
  • Android三种实现定时器的方法
    方法一、使用Handler和Thread(线程)实现定时器 代码如下: package com.xunfang.handerDemo; import android.app....
    99+
    2022-06-06
    方法 定时器 Android
  • Android实现定时器的3种方法
    在Android开发中,定时器一般有以下3种实现方法: 一、采用Handler与线程的sleep(long)方法 二、采用Handler的postDelayed(Runnabl...
    99+
    2022-06-06
    方法 定时器 Android
  • python实现定时器的5种方法
    目录1. 使用time,threading 模块2. 使用datetime,threading 模块3. 使用time,schedule 模块 (执行单个任务)4. 使用time,s...
    99+
    2023-03-06
    python 定时器
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作