0

0

JavaScript多阶段计时器:实现标签切换时计数器重置的技巧

霞舞

霞舞

发布时间:2025-10-09 13:30:14

|

722人浏览过

|

来源于php中文网

原创

JavaScript多阶段计时器:实现标签切换时计数器重置的技巧

本文将指导您如何在JavaScript中构建一个多阶段计时器,特别是在每个阶段(如呼吸练习的不同环节)切换时,如何实现局部计数器自动重置为1。通过引入两个独立的计数变量——一个跟踪整体进度,另一个跟踪当前阶段进度——我们能确保计时器显示符合预期,提供清晰的用户体验。

在开发具有多个顺序阶段的计时器应用时,一个常见的需求是让每个阶段的计数器从1开始重新计时。例如,在一个呼吸练习应用中,可能包含“吸气”、“屏息”、“呼气”、“屏息”等多个环节,每个环节都需要独立的计时显示。如果仅使用一个全局计数器,它将持续递增,无法满足每个阶段独立计时的要求。

问题分析

初始的实现尝试通常会使用一个单一的 count 变量来跟踪时间流逝,并根据 count 的值来更新阶段标签。然而,这种方法导致 count 值在整个计时周期内不断增加,而不是在每个阶段开始时重置为1。

// 原始实现片段(存在问题)
var count = 1;
var interval = setInterval(function() {
  timer.textContent = count; // count持续递增
  if (count <= 8) {
    label.textContent = 'Inhale';
  } else if (count <= 16) { // 此时count已大于8,不会从1开始
    label.textContent = 'Pause Inhale';
  }
  count++;
  // ...
}, 1000);

上述代码的问题在于,timer.textContent 直接显示的是全局 count,当进入下一个阶段时,count 的值已经很大,无法实现从1开始计时的效果。

解决方案:引入独立阶段计数器

要解决这个问题,核心思想是区分两个概念:

  1. 整体进度计数器 (Overall Progress Counter):用于跟踪整个计时周期的总时间,决定何时切换阶段标签。
  2. 阶段内部计数器 (Segment Counter):用于显示当前阶段内部的计时,它会在每个新阶段开始时重置。

我们将使用 count 来表示整体进度,而 segcount 来表示当前阶段的内部进度。

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

实现步骤与代码

1. HTML 结构

首先,我们需要一个简单的HTML结构来显示计时器和当前阶段的标签。

Special Timer

这里,timer 段落用于显示数字计时,label 段落用于显示当前的动作标签(如“Inhale”)。

2. CSS 样式

为了让页面居中显示,我们添加一些基本的CSS样式。

body {
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  height: 100vh;
  margin: 0;
  font-family: Arial, sans-serif;
  text-align: center;
}

h1 {
  font-size: 24px;
  font-weight: bold;
}

p {
  font-size: 18px;
}

3. JavaScript 逻辑

这是实现核心功能的关键部分。我们将修改 startTimer 函数,引入 segcount 变量,并在适当的时候重置它。

问小白
问小白

免费使用DeepSeek满血版

下载
// Function to start the timer
function startTimer() {
  var timer = document.getElementById('timer');
  var label = document.getElementById('label');

  // 初始化第一个阶段的标签
  label.textContent = 'Inhale'; 

  // count 跟踪整个计时周期的总秒数
  var count = 1; 
  // segcount 跟踪当前阶段内部的秒数,会在阶段切换时重置
  var segcount = 1; 

  // Interval for the timer
  var interval = setInterval(function() {
    // 更新计时器显示为当前阶段的计数
    timer.textContent = segcount;

    // 根据整体进度 (count) 判断是否切换阶段标签并重置 segcount
    if (count === 8) { // 8秒吸气结束
      label.textContent = 'Pause Inhale';
      segcount = 0; // 重置segcount,下一秒将变为1
    } else if (count === 16) { // 8秒屏息结束 (总计 8+8=16秒)
      label.textContent = 'Exhale';
      segcount = 0; // 重置segcount
    } else if (count === 28) { // 12秒呼气结束 (总计 16+12=28秒)
      label.textContent = 'Pause Exhale';
      segcount = 0; // 重置segcount
    }
    // 注意:最后一个阶段结束后,segcount 不必重置,因为整个周期即将结束

    // 两个计数器都递增
    count++;
    segcount++;

    // 当整体进度 (count) 达到45时,表示一个完整的呼吸周期(8+8+12+8 = 36秒,
    // 但由于 count 在判断后递增,所以周期结束条件是 36 + 1 = 37,
    // 原始代码中是 45,这里假设原意是 44 秒后停止,所以 45 是停止时的 count 值)
    // 如果是 36秒循环,则应为 count === 37
    // 根据原始代码的逻辑 (8+8+12+8 = 36),count 会从 1 递增到 36,然后变成 37。
    // 如果要完全匹配原代码的 45,则意味着计时总时长为 44 秒。
    // 这里我们沿用原始的 `count === 45` 作为停止条件。
    if (count === 45) { 
      clearInterval(interval); // 清除当前计时器
      startTimer();            // 重新开始一个新的计时周期
    }
  }, 1000); // 每1000毫秒(1秒)执行一次
}

// Start the timer initially
startTimer();

代码详解

  1. 变量初始化:

    • timer 和 label 获取对应的DOM元素。
    • label.textContent = 'Inhale';:在计时器开始时,立即将标签设置为“Inhale”,以便用户知道第一个阶段是什么。
    • count = 1;:count 是一个全局计时器,从1开始,每秒递增一次,用于跟踪整个呼吸周期的总时间。
    • segcount = 1;:segcount 是当前阶段的局部计时器,从1开始,每秒递增一次,用于显示当前阶段的进度。
  2. setInterval 逻辑:

    • timer.textContent = segcount;:每次计时器更新时,timer 元素显示的是 segcount 的值,从而实现了每个阶段从1开始计时的效果。
    • 阶段切换条件:
      • if (count === 8):当 count 达到8时,表示第一个“Inhale”阶段结束。此时,将 label 更新为“Pause Inhale”,并将 segcount 重置为0。为什么是0?因为在 segcount++ 执行后,它会立即变为1,从而在下一个循环周期中显示为1,实现了从1开始计时的效果。
      • else if (count === 16):当 count 达到16(8秒吸气 + 8秒屏息)时,表示“Pause Inhale”阶段结束,切换到“Exhale”,并重置 segcount。
      • else if (count === 28):当 count 达到28(16秒 + 12秒呼气)时,表示“Exhale”阶段结束,切换到“Pause Exhale”,并重置 segcount。
    • count++; 和 segcount++;:在每次循环的最后,两个计数器都递增。
    • 周期结束与重启:
      • if (count === 45):当 count 达到45时,表示一个完整的呼吸周期结束。
      • clearInterval(interval);:停止当前的计时器。
      • startTimer();:重新调用 startTimer() 函数,开始一个新的呼吸周期,从而实现无限循环。

注意事项与最佳实践

  • 明确变量职责: count 负责整体流程控制和阶段判断,segcount 负责当前阶段的显示,这种职责分离是解决问题的关键。

  • 重置时机: segcount 的重置应发生在阶段切换的条件判断内部,并且重置为0是为了在紧随其后的 segcount++ 操作后,能从1开始显示。

  • 可维护性: 对于更复杂的计时器或更多阶段的计时器,可以考虑使用一个数组来存储每个阶段的持续时间及其对应的标签,通过循环遍历数组来管理阶段切换,而不是硬编码多个 if/else if 条件。例如:

    const stages = [
      { label: 'Inhale', duration: 8 },
      { label: 'Pause Inhale', duration: 8 },
      { label: 'Exhale', duration: 12 },
      { label: 'Pause Exhale', duration: 8 }
    ];
    let currentStageIndex = 0;
    let stageStartTime = 0; // 记录当前阶段开始时的总时间
    // ... 在 setInterval 中根据 (count - stageStartTime) 来计算 segcount
    // 并在 (count - stageStartTime) === stages[currentStageIndex].duration 时切换阶段

    这种方式可以使代码更加灵活和易于扩展。

  • 用户体验: 确保计时器显示和标签切换同步,提供清晰的视觉反馈。

总结

通过引入一个专门用于跟踪当前阶段进度的 segcount 变量,并结合一个跟踪整体进度的 count 变量,我们成功解决了多阶段计时器中计数器无法在每个阶段开始时重置的问题。这种分离职责的设计模式不仅适用于呼吸练习,也适用于任何需要分段计时的应用场景,提高了代码的清晰度和功能性。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

557

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

374

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

754

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

478

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

434

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

1031

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

658

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

553

2023.09.20

excel表格操作技巧大全 表格制作excel教程
excel表格操作技巧大全 表格制作excel教程

Excel表格操作的核心技巧在于 熟练使用快捷键、数据处理函数及视图工具,如Ctrl+C/V(复制粘贴)、Alt+=(自动求和)、条件格式、数据验证及数据透视表。掌握这些可大幅提升数据分析与办公效率,实现快速录入、查找、筛选和汇总。

0

2026.01.21

热门下载

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

精品课程

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

共14课时 | 0.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.9万人学习

CSS教程
CSS教程

共754课时 | 21.6万人学习

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

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