js前端如何寫一個(gè)精確的倒計(jì)時(shí)代碼
關(guān)于寫倒計(jì)時(shí)大家可能都都比較熟悉,使用 setTimeout 或 setInterval 就可以搞定。幾秒鐘或者幾分鐘的倒計(jì)時(shí)這樣寫沒有問題,但是如果是長(zhǎng)時(shí)間的倒計(jì)時(shí),這樣寫就會(huì)不準(zhǔn)確。如果用戶修改了他的設(shè)備時(shí)間,這樣的倒計(jì)時(shí)就沒有意義了。今天就說說寫一個(gè)精確的倒計(jì)時(shí)的方法。
原理
眾所周知 setTimeout 或者 setInterval 調(diào)用的時(shí)候會(huì)有微小的誤差。有人做了一個(gè) demo 來觀察這個(gè)現(xiàn)象并對(duì)其做了修正。短時(shí)間的誤差倒也可以接受,但是作為一個(gè)長(zhǎng)時(shí)間的倒計(jì)時(shí),誤差累計(jì)就會(huì)導(dǎo)致倒計(jì)時(shí)不準(zhǔn)確。
因此我們可以在獲取剩余時(shí)間的時(shí)候,每次 new 一個(gè)設(shè)備時(shí)間,因?yàn)樵O(shè)備時(shí)間的流逝相對(duì)是準(zhǔn)確的,并且如果設(shè)備打開了網(wǎng)絡(luò)時(shí)間同步,也會(huì)解決這個(gè)問題。
但是,如果用戶修改了設(shè)備時(shí)間,那么整個(gè)倒計(jì)時(shí)就沒有意義了,用戶只要將設(shè)備時(shí)間修改為倒計(jì)時(shí)的 endTime 就可以輕易看到倒計(jì)時(shí)結(jié)束是頁面的變化。因此一開始獲取服務(wù)端時(shí)間就是很重要的。
簡(jiǎn)單的說,一個(gè)簡(jiǎn)單的精確倒計(jì)時(shí)原理如下:
- 初始化時(shí)請(qǐng)求一次服務(wù)器時(shí)間 serverTime,再 new 一個(gè)設(shè)備時(shí)間 deviceTime
- deviceTime 與 serverTime 的差作為時(shí)間偏移修正
- 每次遞歸時(shí) new 一個(gè)系統(tǒng)時(shí)間,解決 setTimeout 不準(zhǔn)確的問題
代碼
獲取剩余時(shí)間的代碼如下:
/**
* 獲取剩余時(shí)間
* @param {Number} endTime 截止時(shí)間
* @param {Number} deviceTime 設(shè)備時(shí)間
* @param {Number} serverTime 服務(wù)端時(shí)間
* @return {Object} 剩余時(shí)間對(duì)象
*/
let getRemainTime = (endTime, deviceTime, serverTime) => {
let t = endTime - Date.parse(new Date()) - serverTime + deviceTime
let seconds = Math.floor((t / 1000) % 60)
let minutes = Math.floor((t / 1000 / 60) % 60)
let hours = Math.floor((t / (1000 * 60 * 60)) % 24)
let days = Math.floor(t / (1000 * 60 * 60 * 24))
return {
'total': t,
'days': days,
'hours': hours,
'minutes': minutes,
'seconds': seconds
}
}
獲取服務(wù)器時(shí)間可以使用 mtop 接口 mtop.common.getTimestamp
然后可以通過下面的方式來使用:
// 獲取服務(wù)端時(shí)間(獲取服務(wù)端時(shí)間代碼略)
getServerTime((serverTime) => {
//設(shè)置定時(shí)器
let intervalTimer = setInterval(() => {
// 得到剩余時(shí)間
let remainTime = getRemainTime(endTime, deviceTime, serverTime)
// 倒計(jì)時(shí)到兩個(gè)小時(shí)內(nèi)
if (remainTime.total <= 7200000 && remainTime.total > 0) {
// do something
//倒計(jì)時(shí)結(jié)束
} else if (remainTime.total <= 0) {
clearInterval(intervalTimer);
// do something
}
}, 1000)
})
這樣的的寫法也可以做到準(zhǔn)確倒計(jì)時(shí),同時(shí)也比較簡(jiǎn)潔。不需要隔段時(shí)間再去同步一次服務(wù)端時(shí)間。
補(bǔ)充
在寫倒計(jì)時(shí)的時(shí)候遇到了一個(gè)坑這里記錄一下。
千萬別在倒計(jì)時(shí)結(jié)束的時(shí)候請(qǐng)求接口。會(huì)讓服務(wù)端瞬間 QPS 峰值達(dá)到非常高。

如果在倒計(jì)時(shí)結(jié)束的時(shí)候要使用新的數(shù)據(jù)渲染頁面,正確的做法是:
在倒計(jì)時(shí)結(jié)束前的一段時(shí)間里,先請(qǐng)求好數(shù)據(jù),倒計(jì)時(shí)結(jié)束后,再渲染頁面。
關(guān)于倒計(jì)時(shí),如果你有什么更好的解決方案,歡迎評(píng)論交流。
相關(guān)文章
JavaScript中setInterval()和setTimeout()的用法及區(qū)別
這篇文章主要給大家介紹了關(guān)于JavaScript中setInterval()和setTimeout()用法及區(qū)別的相關(guān)資料,Javascript的setTimeOut和setInterval函數(shù)應(yīng)用非常廣泛,它們都用來處理延時(shí)和定時(shí)任務(wù),需要的朋友可以參考下2023-11-11
詳解如何使用JavaScript實(shí)現(xiàn)自定義的雙向數(shù)據(jù)綁定
雙向數(shù)據(jù)綁定是一種編程模式,用于在用戶界面和數(shù)據(jù)模型之間實(shí)現(xiàn)數(shù)據(jù)的同步更新,它允許用戶界面中的數(shù)據(jù)變化自動(dòng)更新到數(shù)據(jù)模型中,在這篇文章中,我會(huì)使用基于觀察者模式和基于Proxy對(duì)象來實(shí)現(xiàn)JS的自定義雙向數(shù)據(jù)綁定2023-08-08
bootstrap自定義樣式之bootstrap實(shí)現(xiàn)側(cè)邊導(dǎo)航欄功能
bootstrap自帶的響應(yīng)式導(dǎo)航欄是向下滑動(dòng)的,有時(shí)滿足不了個(gè)性化的需求,需要做一個(gè)類似于android drawerLayout 側(cè)滑的菜單,這就是我要實(shí)現(xiàn)的bootstrap自定義側(cè)滑菜單。接下來通過本文給大家介紹bootstrap實(shí)現(xiàn)側(cè)邊導(dǎo)航欄功能,感興趣的朋友一起看看吧2018-09-09
Js利用console計(jì)算代碼運(yùn)行時(shí)間的方法示例
最近看了一本書,發(fā)現(xiàn)了個(gè)計(jì)算代碼執(zhí)行時(shí)間的方法,感覺還挺有用的,所以這篇文章主要給大家介紹了關(guān)于Javascript利用console計(jì)算代碼運(yùn)行時(shí)間的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下。2017-09-09
safari,opera嵌入iframe頁面cookie讀取問題解決方法
最近做的合作網(wǎng)站嵌入到對(duì)方的iframe中去,在safari,opera和有些版本的搜狗瀏覽器(內(nèi)核版本原因)中不能讀到cookie。2010-06-06
微信小程序動(dòng)態(tài)顯示項(xiàng)目倒計(jì)時(shí)效果
這篇文章主要為大家詳細(xì)介紹了微信小程序動(dòng)態(tài)顯示項(xiàng)目倒計(jì)時(shí),格式如4天7小時(shí)58分鐘39秒,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06

