0

0

修复 JavaScript 计时器:解决计时不准问题

碧海醫心

碧海醫心

发布时间:2025-11-01 11:24:17

|

733人浏览过

|

来源于php中文网

原创

修复 javascript 计时器:解决计时不准问题

本文旨在解决JavaScript计时器在长时间运行后出现计时不准确的问题。通过分析传统`setTimeout`方法的缺陷,提出使用系统时钟和`requestAnimationFrame`进行优化的方案,并提供详细的代码示例,帮助开发者构建更精确的计时器。

在使用 JavaScript 创建计时器(如秒表)时,开发者可能会遇到一个常见的问题:计时器随着时间的推移会逐渐变得不准确。这是因为传统的计时方法,例如使用 setTimeout,会受到 JavaScript 引擎执行效率、浏览器渲染以及其他因素的影响,导致实际的计时间隔与设定的间隔存在偏差。

问题分析:setTimeout 的局限性

传统的 JavaScript 计时器实现通常依赖于 setTimeout 或 setInterval 函数。这些函数用于在指定的延迟后执行一段代码。然而,它们并非设计用于高精度计时,存在以下局限性:

  1. 执行延迟: setTimeout 设定的延迟时间并非绝对保证。浏览器的事件循环机制会根据当前的任务队列和系统资源情况来决定何时执行回调函数。如果主线程繁忙,回调函数的执行可能会被延迟。
  2. 累积误差: 每次 setTimeout 调用都会引入一定的误差。随着时间的推移,这些误差会累积,导致计时器与实际时间产生明显的偏差。
  3. 浏览器优化: 现代浏览器为了优化性能,可能会对后台标签页中的 setTimeout 调用进行节流,进一步降低计时器的准确性。

解决方案:基于系统时钟的精确计时

为了解决上述问题,可以采用一种基于系统时钟的计时方法。这种方法的核心思想是:

立即学习Java免费学习笔记(深入)”;

  1. 记录起始时间: 在计时器启动时,记录当前的系统时间(例如,使用 Date.now() 或 performance.now())。
  2. 计算时间差: 在每次更新计时器显示时,计算当前时间与起始时间的时间差。
  3. 更新显示: 将时间差转换为小时、分钟、秒和毫秒,并更新计时器的显示。

这种方法避免了依赖 setTimeout 的固定延迟,而是直接根据系统时钟来计算时间差,从而减少了累积误差。

叮当好记-AI音视频转图文
叮当好记-AI音视频转图文

AI音视频转录与总结,内容学习效率 x10!

下载

代码示例

以下是一个使用系统时钟的 JavaScript 秒表示例:

var hr = 0;
var min = 0;
var sec = 0;
var count = 0;

var Startbutton = document.getElementById("Start");

var timer = false;
var start_time;

function start() {
  Startbutton.disabled = true;
  timer = true;
  start_time = new Date().getTime();
  stopwatch();
}

function stop() {
  timer = false;
  start_time = null;
  Startbutton.disabled = false;
}

function reset() {
  timer = false;
  start_time = null;
  Startbutton.disabled = false;
  min = 0;
  hr = 0;
  sec = 0;
  count = 0;

  document.getElementById("hr").innerHTML = "00";
  document.getElementById("min").innerHTML = "00";
  document.getElementById("sec").innerHTML = "00";
  document.getElementById("count").innerHTML = "00";
}

function stopwatch() {
  var now = new Date().getTime();
  var diff = now - start_time;

  if (timer) {
    var str_time = new Date(diff).toISOString().slice(11, 23);

    var hrString = "" + str_time.substring(0, 2);
    var minString = "" + str_time.substring(3, 5);
    var secString = "" + str_time.substring(6, 8);
    var countString = "" + str_time.substring(9, 11);

    document.getElementById("hr").innerHTML = hrString;
    document.getElementById("min").innerHTML = minString;
    document.getElementById("sec").innerHTML = secString;
    document.getElementById("count").innerHTML = countString;
    requestAnimationFrame(stopwatch);
  }
}

代码解释:

  1. start() 函数:
    • 禁用 "Start" 按钮以防止多次点击。
    • 设置 timer 标志为 true 以启动计时器。
    • 使用 new Date().getTime() 记录启动时的系统时间。
    • 调用 stopwatch() 函数开始计时。
  2. stopwatch() 函数:
    • 使用 new Date().getTime() 获取当前时间。
    • 计算当前时间与启动时间的时间差 (diff)。
    • 使用 new Date(diff).toISOString().slice(11, 23) 将时间差格式化为 HH:mm:ss.SSS 字符串。
    • 提取小时、分钟、秒和毫秒部分。
    • 更新 HTML 元素以显示计时器。
    • 使用 requestAnimationFrame(stopwatch) 在下一次浏览器重绘时再次调用 stopwatch() 函数。

使用 requestAnimationFrame 优化

为了进一步提高计时器的准确性和性能,建议使用 requestAnimationFrame 代替 setTimeout。requestAnimationFrame 是一个浏览器 API,用于在浏览器准备好重新绘制屏幕时执行回调函数。它的优点包括:

  1. 与浏览器同步: requestAnimationFrame 的回调函数会在浏览器的下一次重绘之前执行,确保计时器的更新与屏幕刷新同步,避免了画面撕裂和卡顿。
  2. 节能: 当浏览器标签页不可见时,requestAnimationFrame 会自动暂停,从而节省 CPU 资源和电量。
  3. 更高的精度: requestAnimationFrame 通常比 setTimeout 更准确,因为它避免了 setTimeout 的延迟和误差。

在上述代码示例中,requestAnimationFrame(stopwatch) 用于在每次浏览器重绘时调用 stopwatch() 函数,从而实现更平滑和准确的计时效果。

注意事项

  • 选择合适的计时单位: 根据实际需求选择合适的计时单位。如果需要高精度计时,可以使用 performance.now(),它提供更高的分辨率(通常为微秒级别)。
  • 避免阻塞主线程: 在计时器的回调函数中,尽量避免执行耗时的操作,以免阻塞主线程,影响计时器的准确性和用户体验。
  • 测试和校准: 在不同的浏览器和设备上测试计时器,并根据实际情况进行校准,以确保其准确性。

总结

通过使用系统时钟和 requestAnimationFrame,可以构建更精确和高效的 JavaScript 计时器。这种方法避免了传统 setTimeout 的局限性,并充分利用了浏览器的优化机制,从而提供更好的用户体验。希望本文能帮助开发者解决 JavaScript 计时器不准确的问题,并构建更可靠的计时应用。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1570

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

651

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1249

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1206

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

194

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

131

2025.08.07

chatgpt使用指南
chatgpt使用指南

本专题整合了chatgpt使用教程、新手使用说明等等相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 6.1万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 3.5万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号