前端实现一个高精准定时器和延时器的完整代码

 更新时间:2026年03月19日 09:02:26   作者:光影少年  
定时器和延时器是非常常用的功能,用于在指定的时间后执行某些操作,这篇文章主要介绍了前端实现一个高精准定时器和延时器的完整代码,需要的朋友可以参考下

一、为什么浏览器定时器不精准?

JS 是单线程

  • 主线程被占用 → 定时器回调延迟
  • UI / 渲染 / GC 都会阻塞

浏览器最小时间精度限制

  • HTML5 规范限制(4ms)
  • 后台 Tab 被强制降频(1000ms+)

setInterval 的“时间漂移”

setInterval(fn, 1000);
// 实际执行时间 = 1000 + fn执行时间

👉 时间会越来越不准

二、高精准定时器的核心思想(先记住)

不用“执行次数”计时,而用“时间戳”对齐真实时间。

三、高精准延时器(setTimeout 替代方案)

实现:时间戳校正法

function preciseTimeout(fn, delay) {
  const start = performance.now();

  function loop() {
    const now = performance.now();
    if (now - start >= delay) {
      fn();
    } else {
      requestAnimationFrame(loop);
    }
  }

  loop();
}

为什么准?

  • 使用高精度时钟 performance.now()
  • 每帧校正误差

四、高精准定时器(setInterval 替代方案)

实现:递归 setTimeout + 时间修正

function preciseInterval(fn, interval) {
  let expected = performance.now() + interval;

  function step() {
    const drift = performance.now() - expected;

    fn();

    expected += interval;
    setTimeout(step, Math.max(0, interval - drift));
  }

  setTimeout(step, interval);
}

核心点

  • 不是简单等待 interval
  • 每次都修正漂移

五、Web Worker 定时器(更准)

为什么 Worker 更准?

  • 不阻塞 UI 线程
  • 不受页面渲染影响

worker.js

setInterval(() => {
  postMessage(Date.now());
}, 1000);

主线程

const worker = new Worker('worker.js');
worker.onmessage = () => {
  console.log('tick');
};

👉 计时类、音频、监控系统首选

六、requestAnimationFrame 方案(适合动画)

let last = performance.now();

function rafTimer() {
  const now = performance.now();
  if (now - last >= 1000) {
    console.log('tick');
    last = now;
  }
  requestAnimationFrame(rafTimer);
}

rafTimer();

七、三种方案对比(面试必背)

方案精度适合场景
setTimeout普通任务
时间校正业务定时
Web Worker很高长期计时
RAF动画 / UI

八、延时器 vs 定时器(概念区分)

类型特点
延时器执行一次
定时器周期执行
高精准基于时间差

九、后台 / 锁屏问题(高级)

问题

  • 浏览器后台 Tab 会降频

解决方案

  • Web Worker
  • Visibility API + 校正
  • 服务端时间对齐

十、30 秒面试标准回答(直接背)

浏览器原生定时器会因主线程阻塞产生时间漂移;
高精准定时器应基于时间戳校正,而不是固定间隔;
可以通过递归 setTimeout、requestAnimationFrame 或 Web Worker 提升精度;
对精度要求极高的场景应使用 Worker 或服务端时间同步。

到此这篇关于前端实现一个高精准定时器和延时器的文章就介绍到这了,更多相关前端实现定时器和延时器内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • js实现图片粘贴上传到服务器并展示的实例

    js实现图片粘贴上传到服务器并展示的实例

    下面小编就为大家带来一篇js实现图片粘贴上传到服务器并展示的实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-11-11
  • 通过bootstrap全面学习less

    通过bootstrap全面学习less

    这篇文章主要为大家详细介绍了如何通过bootstrap学习less,LESS CSS是一种动态样式语言,属于CSS预处理语言的一种,感兴趣的朋友可以参考下
    2016-11-11
  • JavaScript封装Vue-Router实现流程详解

    JavaScript封装Vue-Router实现流程详解

    这篇文章主要介绍了JavaScript封装Vue-Router实现流程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-09-09
  • JavaScript实现的商品抢购倒计时功能示例

    JavaScript实现的商品抢购倒计时功能示例

    这篇文章主要介绍了JavaScript实现的商品抢购倒计时功能,可实现分秒级别的实时显示倒计时效果,涉及js日期时间计算与页面元素动态操作相关技巧,需要的朋友可以参考下
    2017-04-04
  • javascript实现的简单的表单验证

    javascript实现的简单的表单验证

    这篇文章主要介绍了javascript实现的简单的表单验证的相关资料,需要的朋友可以参考下
    2015-07-07
  • JS是按值传递还是按引用传递

    JS是按值传递还是按引用传递

    在分析这个问题之前,我们需了解什么是按值传递(call by value),什么是按引用传递(call by reference)。在计算机科学里,这个部分叫求值策略(Evaluation Strategy)。它决定变量之间、函数调用时实参和形参之间值是如何传递的。
    2015-01-01
  • 浅析Javascript中“==”与“===”的区别

    浅析Javascript中“==”与“===”的区别

    这篇文章主要介绍了浅析Javascript中“==”与“===”的区别,非常的全面,这里推荐给小伙伴们
    2014-12-12
  • Javascript图片上传前的本地预览实例

    Javascript图片上传前的本地预览实例

    图片的上传预览功能主要用于图片上传前的一个效果的预览,这篇文章主要介绍了Javascript图片上传前的本地预览实例,需要的朋友可以参考下
    2014-06-06
  • SpringMVC restful 注解之@RequestBody进行json与object转换

    SpringMVC restful 注解之@RequestBody进行json与object转换

    这篇文章主要介绍了SpringMVC restful 注解之@RequestBody进行json与object转换的相关资料,需要的朋友可以参考下
    2015-12-12
  • javascript下对于事件、事件流、事件触发的顺序随便说说

    javascript下对于事件、事件流、事件触发的顺序随便说说

    向同一个标签 动态的添加事件是 执行的顺序在ie和其他非ie内核的浏览器有所不同 ie是“先进先出 ” 就是最先添加的最先执行,其他非ie内核的浏览器是 “先进后出”,就是 最后添加的事件 先执行。
    2010-07-07

最新评论