什么?你的 interval 定时器并不靠谱?
yuyutoo 2025-05-25 16:22 1 浏览 0 评论
前言
彦祖们,前端开发中对于 setTimeout setInterval 一定用得烂熟于心了吧?
但你知道你的定时器并没那么靠谱吗?
本文涉及技术栈(非必要)
- vue2
场景复现
今天笔者在开发业务的时候就遇到了这样一个场景
前后端有一个 ws 通道,我们暂且命名为 channel
前后端约定如下:
- 前端每隔 5000ms 发送一个 ping 消息
- 后端收到 ping 后回复一个 pong 消息
- 后端如果 15000ms 未收到 ping,则视为 channel 失活,直接 kill
- kill 后前端会主动发起重连
文章还没写两分钟,一只暴躁的测试老哥说道:"你们的 ws 也太不稳定了,几十秒就断开一次?废物?"
骂骂咧咧的甩过来一张截图
笔者心想:"为什么我的界面稳如老狗?浏览器问题,绝对是浏览器问题..."
起身查看,遂发现毫无问题,和笔者一模一样的 chrome 版本...
静心而坐,对着浏览器屏幕茶颜悦色(哦,察言观色)...
10 分钟过去了,半小时过去了...还是稳如老狗,根本不断
问题分析
那么问题到底出在哪里呢?
笔者坐在测试妹纸身边仔细观察了她的操作后!
发现她不断得切屏,此时已初步心虚,不禁问道 GPT
当浏览器标签页变为非活动状态时,setInterval 和 setTimeout 的执行频率通常会被降级。大多数现代浏览器将其执行频率限制在 1 秒(1000 毫秒)或更高,以减少 CPU 和电池的消耗。
问题原因大致是这样了
问题复现
此时笔者在本地写了个 demo
理想的情况,我们这个 offset 应该是一直维持在 1000ms 左右
那么后续我们就要看页面激活 | 失活时候的情况了
页面激活时
我们先看下页面激活时的打印数据
没什么问题,符合我们的期望值
页面失活时
接下来我们,切换到其他浏览器标签,保持几分钟,几分钟后我们看下打印数据
明显发现有些数据不符合我们的期望值
甚至有些夸张到长达 41003ms,将近 40 倍,不靠谱!
寻找方案
用 setTimeout 模拟 setInterval
其实网上最多的方案就是说用 setTimeout 模拟 setInterval
但是很可惜,笔者亲自模拟下来,也是同样的结果,我们看截图
而且发现更加不靠谱了...错误的概率明显更高了...
其实可想而知,setInterval 和 setTimeout 在事件循环中都属于 Task
事件循环的优先级是一样的,同样都属于主线程任务(标记起来,后面考重点)
Web Worker
其实网上还有类似于 requestAnimationFrame 的方案
但是测试下来更离谱,就不浪费彦祖们的时间了
进入正题吧
其实上文说了,主线程任务的优先级会被降低,那么我们思考一下子线程任务呢?
子线程任务在前端领域,我们不就能想到 Web Worker 吗?
当然除了 Web Worker,还有 SharedWorker Service Worker
非本文重点,不做赘述
什么是 Web Worker
首先我们来认识下什么是 Web Worker
Web Worker 是一种运行在浏览器后台的独立 JavaScript 线程,允许我们在不阻塞主线程(即不影响页面 UI 和用户交互的情况下)执行一些耗时的任务,比如数据处理、文件操作、复杂计算等。
不阻塞主线程这恰恰是我们的所需要的!
使用 Web Worker
其实 Web Worker 在常规使用和 vue 中还是有一定的区别的
常规使用
常规使用其实非常简单,我们还是以上文中的 demo 为例
改造一下
- index.html
我们还需要一个 worker.js文件
- worker.js
切换 tab 几分钟后让我们来看看打印结果
非常完美,几乎都保持在 1000ms 左右
在 vue 中使用 Web Worker
在 vue 中使用就和常规使用有所不同了
这也是笔者今天踩坑比较多的地方
网上很多文中配置了 webpack 的 worker-loader,然后改造 vue.config.js
但是笔者多次尝试,还是各种报错(如果有大佬踩过坑,请在评论区留言)
最后笔者翻到了之前的笔记,其实早在多年之前就记录了在 vue 中使用 Web Worker 的文章
使用方式非常简单
我们只需要把 worker.js 放置于 public 目录即可!
看下我们此时的代码
- App.vue
- public worker.js
测试一下
非常完美!
解决业务问题
彦祖们此时可能要问道,你只是证明了 Web Worker 不会阻塞主进程
和你的业务有什么关系吗?
其实这还得依赖于Web Worker的通信机制
我们继续改造
- App.vue
- worker.js
封装一个 setWorkerInterval
其实有了以上的代码模型,我们就能封装一个不受主进程阻塞的定时器了
我们暂且命名它为 setWorkerInterval
函数设计
首先设计一下我们的函数
为了减少开发者心智负担
我们需要把函数设计成和 setInterval 一样的用法
我们在使用 setInterval 的时候,日常最常用的参数就是 callback 和 delay
它的返回值是一个 intervalID
由此可见我们的函数签名如下
动手实现
有了上面的函数设计,我们就开始来实现
目前我们遇到一个问题,那就是上文中的 xxx 具体是个啥?
这其实就是 Web Worker 中的 setInterval
我们只需要把 Web Worker 中的 setInterval的功能暴露给主线程不就完事了吗?
来看 代码
- setWorkerInterval.js
- worker.js
这样我们就初步完成了以上 xxx 的逻辑
但随之而来又有两个问题
1.如何触发对应业务逻辑?
2.如何清除定时器?
触发对应业务逻辑
其实第一个问题非常容易解决,我们不是传递了一个 callback 吗?
这不就是我们的业务逻辑吗
改造一下
- setWorkerInteraval.js
清除定时器
第二个问题也不难,我们思考 intervalID 的来源不就在 worker.js吗?
那我们只需要把它通知给主线程即可,直接上代码
- worker.js
- setWorkerInteraval
让我们看下使用方式
非常完美,至此,一个靠谱的定时器我们就完成了!
当然我们还可以把上文中的 1000ms 改成 delay 传参,直接看完成代码吧
完整代码
- worker.js (vue项目 需要放在 public 中)
- setWorkerInteraval
原文链接:
https://juejin.cn/post/7418391732163182607
相关推荐
- 12、高阶组件:魔法增幅器——React 19 HOC模式
-
一、魔法增幅器的本质"高阶组件是魔法师用咒语叠加的炼金术,"霍格沃茨魔咒研究院院长凝视着发光的增幅器,"通过函数式能量场的嵌套,让基础组件获得预言家日报式的逻辑继承!"...
- 深入理解nodejs的异步IO与事件模块机制
-
一、node为什么要使用异步I/O异步最先诞生于操作系统的底层,在底层系统中,异步通过信号量、消息等方式有广泛的应用。但在大多数高级编程语言中,异步并不多见,这是因为编写异步的程序不符合人习惯的思维逻...
- 前端时间同步利器:React + useEffect 实现高性能动态时钟
-
前言在你奋笔疾敲代码的瞬间,是不是突然一低头,发现时间像偷偷跑路的变量,一眨眼就从上午飘到下午?饭没吃、会没开、工位也快被前端猫霸占了。仿佛你写的不是代码,而是“时间穿梭机”。别慌,咱们今天就来用R...
- JavaScript 异步编程指南 - 聊聊 Node.js 中的事件循环
-
作者:五月君来源:编程界|事件循环是一种控制应用程序的运行机制,在不同的运行时环境有不同的实现,上一节讲了浏览器中的事件循环,它们有很多相似的地方,也有着各自的特点,本节讨论下Node.js中...
- 10个Vue开发技巧「实践」
-
作者:WahFung转发链接:https://juejin.im/post/5e8a9b1ae51d45470720bdfa路由参数解耦一般在组件内使用路由参数,大多数人会这样做:...
- 通过番计时器实例学习 React 生命周期函数 componentDidMount
-
大家好,今天我们将通过一个实例——番茄计时器,学习下如何使用函数生命周期的一个重要函数componentDidMount():componentDidMount(),在组件加载完成,render之后...
- 前端必看!10 个 Vue3 救命技巧,解决你 90% 的开发难题?
-
写Vue3项目时,是不是总被数据更新延迟、组件间传值混乱、页面加载缓慢这些问题折磨得头秃?别担心!作为摸爬滚打多年的老前端,今天掏出压箱底的10个实战技巧,从性能优化到复杂逻辑处理,每一个都能...
- 如何用2 KB代码实现3D赛车游戏?2kPlus Jam大赛了解一下
-
选自frankforce作者:Frank机器之心编译参与:王子嘉、GeekAI控制复杂度一直是软件开发的核心问题之一,一代代的计算机从业者纷纷贡献着自己的智慧,试图降低程序的计算复杂度。然而,将一款...
- 证明你访问的网站是你想访问的,Safari 真的需要
-
安全研究员在Safari上找到了一个新漏洞,能让网站在浏览器的地址栏内将自己伪装成另一个网站——得益于Safari地址栏的“智能缩略”功能。在Deusen最近公开的攻击演示(PoC,P...
- 抓狂!TS 组件性能拉胯到崩溃?4 个绝杀技巧逆风翻盘!
-
前端兄弟姐妹们五一假期快乐,咱们谁还没被TypeScript组件的性能问题折磨过?页面加载转圈圈,点击按钮没反应,代码改了一轮又一轮,性能却还是原地踏步,分分钟想砸电脑!别慌,今天这4个绝杀技...
- 让小球做圆周运动,你有几种办法?
-
最近在阅读外国技术文章中无意中发现了一个神奇的CSS属性motion-path,它可以让Dom元素可以按照自定义的路径移动。又想起了很久之前参加校招面试的时候,面试官问了我一个问题“能不能不借助库实现...
- 前端基础进阶(十四):深入核心,详解事件循环机制
-
EventLoopJavaScript的学习零散而庞杂,很多时候我们学到了一些东西,但是却没办法感受到进步!甚至过了不久,就把学到的东西给忘了。为了解决自己的这个困扰,在学习的过程中,我一直在试图寻...
- 从0搭建一个WebRTC,实现多房间多对多通话,并实现屏幕录制
-
这篇文章开始会实现一个一对一WebRTC和多对多的WebRTC,以及基于屏幕共享的录制。本篇会实现信令和前端部分,信令使用fastity来搭建,前端部分使用Vue3来实现。为什么要使用WebRTCWe...
- Vue2 开发卡壳?这 10 个实战技巧专治各种不服
-
干前端开发的兄弟,谁还没被Vue2折腾过?数据不更新、组件通信乱成麻、性能差到想砸电脑……这些痛点,我都懂!今天直接甩出10个超实用的实战技巧,每一个都是从项目“血坑”里爬出来总结的,专...
你 发表评论:
欢迎- 一周热门
-
-
前端面试:iframe 的优缺点? iframe有那些缺点
-
带斜线的表头制作好了,如何填充内容?这几种方法你更喜欢哪个?
-
漫学笔记之PHP.ini常用的配置信息
-
其实模版网站在开发工作中很重要,推荐几个参考站给大家
-
推荐7个模板代码和其他游戏源码下载的网址
-
[干货] JAVA - JVM - 2 内存两分 [干货]+java+-+jvm+-+2+内存两分吗
-
正在学习使用python搭建自动化测试框架?这个系统包你可能会用到
-
织梦(Dedecms)建站教程 织梦建站详细步骤
-
【开源分享】2024PHP在线客服系统源码(搭建教程+终身使用)
-
2024PHP在线客服系统源码+完全开源 带详细搭建教程
-
- 最近发表
-
- 12、高阶组件:魔法增幅器——React 19 HOC模式
- 深入理解nodejs的异步IO与事件模块机制
- 前端时间同步利器:React + useEffect 实现高性能动态时钟
- JavaScript 异步编程指南 - 聊聊 Node.js 中的事件循环
- 10个Vue开发技巧「实践」
- 通过番计时器实例学习 React 生命周期函数 componentDidMount
- SRE监控四大黄金指标,任何一个有异常都会是灾难……
- 前端必看!10 个 Vue3 救命技巧,解决你 90% 的开发难题?
- 如何用2 KB代码实现3D赛车游戏?2kPlus Jam大赛了解一下
- 证明你访问的网站是你想访问的,Safari 真的需要
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)