实现一个高性能倒计时:从踩坑到最佳实践

前言

本次来分享一个在基于react18中是实现一个"按钮触发倒计时"的完整思路和优化实践。

使用定时器的弊端

setInterval 或者 setTimeout 定时器实现倒计时是一个比较常方案,但是有一个无法避免的 "坑",在浏览器中,如果用户切换到其它的tab页签后,定时器就会不准确

使用 rAF 实现

requestAnimationFrame有以下几个好处:

  • rAF 与浏览器渲染节奏同步,动画/计时更平滑
  • rAF 提供每帧时间戳,易做精度与边界控制

核心思路(Hooks + rAF)

  • 使用 useRef 存放"可变但不触发渲染"的数据

    • rafIdRef:当前动画帧 id
    • endTimeRef:结束时间戳(毫秒)
    • lastSecondRef:上一次已渲染的秒数
  • 仅在秒数变化时 setState(用 lastSecondRef 去重),因为 rAF 每秒可触发 60 次渲染,过多占用cpu

  • 启动新倒计时前,取消旧的 rAF,避免并发

  • 组件卸载时取消 rAF,防泄漏

  • Math.ceil(diff / 1000) 处理边界更稳。Math.ceil 在接近 0 的边界,ceil 更符合"剩余秒数"的用户心智,不会提前跳到更小的值

核心代码

scss 复制代码
import React, { useRef, useState, useEffect } from 'react'

function App() {
  const [count, setCount] = useState(5)

  const rafIdRef = useRef(null)
  const endTimeRef = useRef(0)
  const lastSecondRef = useRef(null)

  const startCountdown = (durationSec) => {
    // 1) 防并发:先取消已有动画帧
    if (rafIdRef.current !== null) {
      cancelAnimationFrame(rafIdRef.current)
      rafIdRef.current = null
    }

    // 2) 初始化时间数据
    const start = performance.now()
    endTimeRef.current = start + durationSec * 1000
    lastSecondRef.current = null

    // 3) rAF 驱动:只在秒数变化时 setState
    const step = (ts) => {
      const diff = endTimeRef.current - ts
      if (diff > 0) {
        const sec = Math.ceil(diff / 1000)
        // 关键优化,避免重复过渡的 setState
        if (sec !== lastSecondRef.current) {
          lastSecondRef.current = sec
          setCount(sec)
        }
        rafIdRef.current = requestAnimationFrame(step)
      } else {
        setCount(0)
        rafIdRef.current = null
      }
    }
    rafIdRef.current = requestAnimationFrame(step)
  }

  // 4) 卸载清理,防止泄漏
  useEffect(() => {
    return () => {
      if (rafIdRef.current !== null) {
        cancelAnimationFrame(rafIdRef.current)
      }
    }
  }, [])

  return (
    <div>
      <p>剩余秒数:{count}</p>
      <button onClick={() => startCountdown(5)}>开始 5 秒倒计时</button>
    </div>
  )
}

export default App

结语

如果这篇文章帮到了你,欢迎点赞👍和关注⭐️。

文章如有错误之处,希望在评论区指正🙏🙏。

相关推荐
张3蜂1 天前
Python 四大 Web 框架对比解析:FastAPI、Django、Flask 与 Tornado
前端·python·fastapi
南风知我意9571 天前
【前端面试5】手写Function原型方法
前端·面试·职场和发展
qq_12498707531 天前
基于Java Web的城市花园小区维修管理系统的设计与实现(源码+论文+部署+安装)
java·开发语言·前端·spring boot·spring·毕业设计·计算机毕业设计
摘星编程1 天前
用React Native开发OpenHarmony应用:Image网络图片加载
javascript·react native·react.js
摘星编程1 天前
OpenHarmony环境下React Native:ImageBase64图片显示
javascript·react native·react.js
阿蒙Amon1 天前
TypeScript学习-第13章:实战与最佳实践
javascript·学习·typescript
小安驾到1 天前
【前端的坑】vxe-grid表格tooltip提示框不显示bug
前端·vue.js
去码头整点薯条981 天前
python第五次作业
linux·前端·python
沐墨染1 天前
Vue实战:自动化研判报告组件的设计与实现
前端·javascript·信息可视化·数据分析·自动化·vue
奔跑的呱呱牛1 天前
viewer-utils 图片预览工具库
javascript·vue·react