当前位置: 首页 > news >正文

如何解决js定时器不准确问题

为什么会出现定时器不准确呢?

        这个其实就得提到js执行机制了,叫做事件循环Eventloop 循环机制中,异步事件 setInterval 到时后会把回调函数放入消息队列中Event Queue,主线程的宏任务执行完毕后依次执行消息队列的微任务,等微任务执行完了在循环回来执行宏任务。并且由于消息队列中存在大量任务,其他任务执行时间就会造成定时器回调函数的延迟,如果不处理则会一直叠加延迟。

以下是ChatGPT给出的的一些解决JS定时器不准确问题的方法:

  1. 使用精度更高的定时器:使用requestAnimationFrame()代替setInterval或setTimeout,因为它以常规更新率刷新屏幕,并保证在用户可见的时间内绘制动画。requestAnimationFrame()是基于浏览器屏幕的重绘机制触发的,可以有效避免误差的累积。

  2. 缩短定时间隔时间:如果一个定时器在浏览器里表现得很不准确,可能需要缩小时间间隔,比如说从100毫秒改成10毫秒。

  3. 使用标准时间: 可以使用标准时间对象提供的函数(如getTime、getSeconds等)获取当前时间,而不是使用JavaScript自带的全局变量Date来保证计时器的准确性。

  4. 避免多个计时器同时存在:如果多个计时器同时存在,可能会导致其他定时器的执行被延迟或丢失调用。只使用唯一一个计时器进行安排和管理。

        总之,实际场景中,可结合具体应用场景选择相符的解决方案。

        我们开发过程中也可以通过计算时差可以有效的解决。


        接下来是我自己整理的方法


 通过动态计算时差解决setInterval定时器不准确的问题

        根据定时器最开始时间计算当前时间(回调函数执行时间)与开始时间的误差,用期望时差减误差作为下一次任务的时间间隔

  1. 在开始执行计时器之前,记录本地时间。

  2. 在计时器函数中,获取当前本地时间,并计算与记录的本地时间之间的时差(单位为毫秒)。

  3. 在计算得到的时差的基础上,添加上期望的时间间隔,便是下一次计时器应该触发的时间。

  4. 在定时器回调函数中,采用setTimeout代替setInterval,并传入计算得到的时间差作为等待时间。

let localTime = new Date().getTime(); //记录本地时间
function timer() {const now = new Date().getTime(); // 获取当前本地时间const timeGap = now - localTime; // 计算时间差// 下一次计时器应该触发的时间const nextTickTime = 1000 - (timeGap % 1000);setTimeout(() => {// 在此处执行计时器的操作console.log('tick');timer(); // 递归调用,实现循环}, nextTickTime);
}timer(); // 启动计时器

      这种方法能够避免JavaScript在处理大量任务时的卡顿和延迟,保证定时器的调用的准确性。

使用 web Worker解决setInterval定时器不准确的问题

         原理:将定时函数作为独立线程执行

        Web Worker 是运行在后台线程中的 JavaScript 脚本,可以与主线程并行工作,因此不会受主线程的影响。我们可以使用 Web Worker 来创建一个新的线程来处理定时器。

以下是一个简单的示例:

//在 main.js 中创建 Worker
const worker = new Worker("worker.js");//在 worker.js 中处理定时器
let intervalId = null;
worker.onmessage = function(event) {console.log("Received message from main script");if (event.data === "start") {intervalId = setInterval(function() {console.log("Worker: Interval fired");postMessage("tick");}, 1000);} else if (event.data === "stop") {clearInterval(intervalId);intervalId = null;}
};

        在主线程中,我们首先创建一个新的 Web Worker worker.js,然后通过调用 onmessage 方法来监听来自 Worker 的消息。当收到 start 消息时,我们在 Worker 中启动一个定时器。当收到 stop 消息时,我们清除定时器。

以下是 worker.js 文件的内容:

//worker.js
onmessage = function(event) {console.log("Received message from main script");if (event.data === "start") {console.log("Worker: Starting interval");intervalId = setInterval(function() {console.log("Worker: Interval fired");postMessage("tick");}, 1000);} else if (event.data === "stop") {console.log("Worker: Stopping interval");clearInterval(intervalId);intervalId = null;}
};

        在 Worker 中,我们定义了一个 onmessage 方法来监听来自主线程的消息。当收到 start 消息时,我们在 Worker 中启动一个定时器;当收到 stop 消息时,我们清除定时器。通过调用 postMessage 方法来将消息发送回主线程。

        现在,可以通过向 Worker 发送 startstop 消息来控制定时器的启动和停止。由于该定时器是在 Worker 线程中运行的,因此它不会受到主线程的影响,从而保证了定时器的准确性。

相关文章:

如何解决js定时器不准确问题

为什么会出现定时器不准确呢? 这个其实就得提到js执行机制了,叫做事件循环Eventloop 循环机制中,异步事件 setInterval 到时后会把回调函数放入消息队列中Event Queue,主线程的宏任务执行完毕后依次执行消息队列的微任务&#xff…...

学习笔记——vue中使用el-dropdown组件报错

今天在工作中,发现使用el-select做的下拉框,下拉菜单展开后,鼠标点击下拉框之外的区域时,下拉菜单没有收起。然后,我打开控制台,发现了这个错误。 Uncaught TypeError: Cannot read properties of null (re…...

Java之旅(八)

Java 条件运算符 Java 条件运算符用于根据一个条件表达式的布尔值来决定程序执行的流程。条件运算符有三种类型:if、else 和 switch。 if 语句的一般格式如下: if (condition) {// 条件为 true 执行的代码块 } 其中,condition 是一个 bool…...

华为OD机试真题(Java),四则运算(100%通过+复盘思路)

一、题目描述 输入一个表达式(用字符串表示),求这个表达式的值。 保证字符串中的有效字符包括[‘0’-‘9’],‘+’,‘-’, ‘*’,‘/’ ,‘(’, ‘)’,‘[’, ‘]’,‘{’ ,‘}’。且表达式一定合法。 数据范围:表达式计算结果和过程中满足∣val∣≤1000 ,字符串长度满…...

HTML表单标签form分析

说明:在html的标签中,表单标签与后台联系密切,像用户登录、注册,都是用到页面的表单标签,用户将信息填入到表单中,提交到后端业务中校验处理,再将结果反馈给前端页面。 表单内的标签分别有&…...

Qt 项目文件Pri详解

在Qt项目中,pri文件(.pri)是一种类似于makefile的文件,用于定义Qt项目中的编译规则。通常可以用pri文件来配置Qt库、头文件、源文件、链接库等信息,这样可以把这些信息定义在一个文件中,避免在每个工程中都进行重复配置&#xff0…...

Keil 5 MDK 发律师函警告了,如何用STCubeIDE开发标准库的程序(STM32F103C8T6为例)

用STCubeIDE进行标准库开发 1、CubeIDE介绍 https://www.stmcu.com.cn/ecosystem/Cube/STM32CubeIDE 2、CubeIDE下载 点击上面的链接,登录即可下载 3、搭建Demo工程 新建一个工作空间 创建一个工程 选择芯片-STM32F103C8T6 填写工程信息 添加标准库到工程 标…...

接口测试--apipost接口断言详解

在做接口测试的时候,会对接口进行断言,一个完整的接口测试,包括:请求->获取响应正文->断言。 一、apipost如何进行断言 apipost的断言设置实在后执行脚本中进行编写的。apipost本身提供了11中断言: apt.asser…...

YYDS练手 130道python练习题 完整版PDF

近年来,Python在编程语言界里赚足了风头,无论是受欢迎程度,还是薪资待遇,都非常可观,相应的,Python岗位要求也越来越高,无论你是零基础还是老前辈,在Python面试中都不能轻视。 不打…...

2-python的变量类型

内容提要 主要回顾了python中的变量类型,区分它们和c中的区别 python中的数字没有long python中的字符串不能被改变,也就是说不能对其内字符元素进行赋值操作 python中列表的数据类型与c访问方式有一定的区别列表的下标可以是负数,无论正负&…...

Python之并发编程一背景知识

一、开篇介绍 顾名思义,进程即正在执行的一个过程。进程是对正在运行程序的一个抽象。 进程的概念起源于操作系统,是操作系统最核心的概念,也是操作系统提供的最古老也是最重要的抽象概念之一。操作系统的其他所有内容都是围绕进程的概念展…...

Redis分区

分区 Redis是单线程的,如何提高多核CPU的利用率? 可以在同一个服务器部署多个Redis的实例,并把他们当作不同的服务器来使用,在某些时候,无论如何一个服务器是不够的, 所以,如果你想使用多个CPU&…...

代码随想录算法训练营第56天 | 583、72

583. 两个字符串的删除操作 题目描述 给定两个单词 word1 和 word2 ,返回使得 word1 和 word2 相同所需的最小步数。 每步 可以删除任意一个字符串中的一个字符。 示例1: 输入: w o r d 1 " s e a " , w o r d 2 " e a t …...

c++输入输出文件操作stream

系列文章目录 C IO库 文章目录 系列文章目录前言一、文件IO概述coutcin其他istream类方法 文件输入和输出内核格式化总结 前言 一、文件IO 概述 c程序把输入和输出看作字节流。输入时,程序从输入流中抽取字节:输出时,程序将字节流插入到输…...

【小呆的力学笔记】非线性有限元的初步认识【三】

文章目录 1.2.2 基于最小势能原理的线性有限元一般格式1.2.2.1 离散化1.2.2.2 位移插值1.2.2.3 单元应变1.2.2.4 单元应力1.2.2.5 单元刚度矩阵1.2.2.6 整体刚度矩阵1.2.2.7 处理约束1.2.2.8 求解节点载荷列阵1.2.2.9 求解位移列阵1.2.2.10 计算应力矩阵等 1.2.2 基于最小势能原…...

python计算闰年

这里说明一下:看到网上很多写python计算闰年的,有很多是不同。 有份省级期刊万年历计算公元1-10000年的闰年 算法如下:4000年停闰一次。区别其他算法,有些是3200年停闰一次。 def division(dividend, divisor) -> bool:"…...

聊聊如何使用Js写一个简单的二级联动和三级联动呢?

前言:咳咳哈,大佬说:"这不是有手就行了?"好吧,这里不做过多罗里吧嗦,真的不过多吹,我们在下面直接上代码上注释。 文章目录: 原Js二级联动实现原Js三级联动实现 一、二级…...

IPv4 和 IPv6 的特点、区别以及在互联网中的应用

在当今互联网时代,IP 地址是连接和通信的基础。IPv4(Internet Protocol version 4)和 IPv6(Internet Protocol version 6)是两种常见的 IP 地址版本。IPv4 是最早广泛使用的 IP 地址协议,而 IPv6 则是 IPv4…...

JavaScript处理移动web交互

touch对象和touchevent touch事件 touch对象 每一次发生touch事件时就会产生一个touch对象&#xff0c;类似事件处理函数中的事件对象。 <div class" "><button class"child" style"height: 400px; width: 400px">我是按钮</b…...

4年测试经验,一问三不知,过于离谱...

公司今年要招人&#xff0c;面倒是面了很多测试&#xff0c;但没有一个合适的。一开始想要的就是中级的水准&#xff0c;也没指望来大牛&#xff0c;当然来了更好&#xff0c;提供的薪资在10-20k,来面试的人有很多&#xff0c;但平均水准真的是让人失望。 看简历时很多都写着3…...

如何用Lingui.js在SSG项目中实现完美国际化:终极指南

如何用Lingui.js在SSG项目中实现完美国际化&#xff1a;终极指南 【免费下载链接】js-lingui &#x1f30d; &#x1f4d6; A readable, automated, and optimized (2 kb) internationalization for JavaScript 项目地址: https://gitcode.com/gh_mirrors/js/js-lingui …...

基于stm32的楼道照明系统[单片机]-计算机毕业设计源码+LW文档

摘要&#xff1a;本文提出了一种基于STM32单片机的楼道照明系统设计方案。该系统以STM32为核心控制器&#xff0c;结合人体热释电感应模块、声音感应模块和光照检测模块&#xff0c;实现楼道照明的智能控制。通过实时检测人体存在、声音信号以及环境光照强度&#xff0c;系统能…...

2026年最新codex 第三方 api 配置指南

真正决定 Codex 能不能顺利进入项目的&#xff0c;通常不是 npm 命令有没有跑完&#xff0c;而是 codex 第三方 api 是否配完整。很多人在 openai/codex 安装结束后马上就碰到 401、请求超时、模型不可用&#xff0c;甚至一直过不了认证&#xff0c;这类问题大多都落在 ~/.code…...

手把手教你用Copilot插件在Obsidian里免费接入DeepSeek-R1(附硅基流动API密钥获取)

零成本解锁Obsidian智能助手&#xff1a;DeepSeek-R1全流程实战指南 在信息爆炸的时代&#xff0c;如何让个人知识管理工具具备AI思维能力&#xff0c;已成为数字笔记用户的核心诉求。Obsidian作为一款以本地优先为理念的Markdown笔记工具&#xff0c;其插件生态正逐步融入大语…...

2026届毕业生推荐的十大降重复率神器解析与推荐

Ai论文网站排名&#xff08;开题报告、文献综述、降aigc率、降重综合对比&#xff09; TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 降低AIGC痕迹的关键之处在于去除机器生成的那种模式化特性&#xff0c;如果要采用避免使用过…...

解锁Windows效率提升:免费工具Winhance-zh_CN全功能指南

解锁Windows效率提升&#xff1a;免费工具Winhance-zh_CN全功能指南 【免费下载链接】Winhance-zh_CN A Chinese version of Winhance. C# application designed to optimize and customize your Windows experience. 项目地址: https://gitcode.com/gh_mirrors/wi/Winhance-…...

AI Memory 全景解析:让 Agent 真正记住你

AI Memory 全景解析&#xff1a;让 Agent 真正"记住"你 你有没有遇到过这种场景&#xff1a;明明昨天告诉 AI 助手你喜欢简洁的代码风格&#xff0c;今天它又开始写冗长的注释&#xff1b;或者你费心纠正了一个错误&#xff0c;下次对话它照犯不误。这就是 AI 没有记…...

从SAP实施到微信上线:一文读懂不同类型软件公司的实施岗到底有啥区别

从SAP实施到微信上线&#xff1a;一文读懂不同类型软件公司的实施岗核心差异 刚入行的技术新人小张最近很困惑&#xff1a;同样是"软件实施工程师"岗位&#xff0c;为什么招聘JD里有的要求精通SAP模块配置&#xff0c;有的却强调微信生态部署经验&#xff1f;在面试了…...

解决Python文件路径超长问题:Windows系统下的终极指南

解决Python文件路径超长问题&#xff1a;Windows系统下的终极指南 在Windows平台上开发Python应用时&#xff0c;文件路径长度限制是个令人头疼的"历史遗留问题"。记得第一次接手一个大型Python项目时&#xff0c;我花了整整两天时间才搞明白为什么某些文件总是无法读…...

给 Claude Code 装上浏览器:Chrome 集成测试版详解

程序员们早就习惯了在终端里跟 AI 助手聊天、改代码、跑测试。但有一个场景始终有点绕——代码改完了&#xff0c;得切到浏览器里看看效果、查查报错、填填表单&#xff0c;然后再切回终端告诉 AI “好像还差点意思”。来回折腾几次&#xff0c;思路容易断。 Anthropic 最近放出…...