<p>最稳妥方案是用uView CountDown组件+服务端毫秒级结束时间戳锚定,每次UI更新前重算Math.max(0, Math.floor((endTimestamp - Date.now()) / 1000)),300ms刷新一次,禁用累减逻辑以防跨端不准。</p>
直接用 uView 的 CountDown 组件 + 服务端时间戳锚定,是最稳妥的方案。别自己封装“减法倒计时”,跨端必不准。
系统后台休眠、iOS 节流、H5 切页、小程序 setData 延迟,都会让累减值脱离真实时间。比如:用户切到微信聊天再切回来,countdown 可能卡在 27 不动,或直接跳到 -3;Android 省电模式下 setTimeout 延迟可达 2–5 秒,累减逻辑完全失效。
end_timestamp: 1743165680000
Math.max(0, Math.floor((endTimestamp - Date.now()) / 1000))
timestamp 属性做倒计时,就别手动改它——让它只由服务端时间驱动uView 的 CountDown 默认支持服务端时间戳驱动,但很多人传错类型或漏掉关键参数,导致组件静默不更新。
:time="remainingSeconds",其中 remainingSeconds 是动态计算出的整数,且需响应式更新@finish="handleFinish" 处理结束逻辑,别只靠 v-if="remainingSeconds > 0" 控制按钮显隐show-days 为 true,且 remainingSeconds >= 86400,否则组件自动隐藏该字段(这是它的智能逻辑,不是 bug)唯品会类抢购页面最常崩在“用户狂点发送”和“倒计时结束瞬间多个请求并发”。这不是 UI 问题,是状态没闭环。
isCounting: false 和 isSubmitted: false 都要放在 data 里if (this.isCounting || this.isSubmitted) return
this.isCounting = true,并禁用按钮;成功/失败回调里必须调用统一重置函数 this.resetCountdown()
resetCountdown() 必须同时处理:this.remainingSeconds = 0、this.isCounting = false、this.isSubmitted = false,缺一不可setTimeout 递归),onShow 中不自动恢复,而是重新拉取最新 end_timestamp 再算一次鸿蒙系统对 setTimeout 的最小间隔限制更严(部分机型 ≥ 500ms),iOS 在后台会冻结 JS 线程,这两个平台会让“每秒触发”变成“每 3 秒触发一次”,UI 明显卡顿。
setInterval,哪怕只用于 UI 刷新——uView CountDown 内部已用 setTimeout 递归实现,你只需保证 time 值实时准确performance.now()(不支持);一律用 Date.now()
onShow 中必须立刻重算 remainingSeconds,不能等下一个定时器 tick —— 否则用户看到的是“假的 00:00:05”,实际活动已开始 3 秒