Javascript實(shí)現(xiàn)商品秒殺倒計(jì)時(shí)(時(shí)間與服務(wù)器時(shí)間同步)
現(xiàn)在有很多網(wǎng)站都在做秒殺商品,而這其中有一個(gè)很重要的環(huán)節(jié)就是倒計(jì)時(shí)。
關(guān)于倒計(jì)時(shí),有下面幾點(diǎn)需要注意:
1.應(yīng)該使用服務(wù)器時(shí)間而不是本地時(shí)間(本地時(shí)間存在時(shí)區(qū)不同、用戶自行設(shè)置等問題)。
2.要考慮網(wǎng)絡(luò)傳輸?shù)暮臅r(shí)。
3.獲取時(shí)間時(shí)可直接從AJAX的響應(yīng)頭中讀?。ㄍㄟ^getResponseHeader('Date')來獲得),服務(wù)器端不需要專門寫時(shí)間生成腳本。
過程分析:
1.從服務(wù)器讀到一個(gè)時(shí)間戳之后便開始計(jì)時(shí),不考慮網(wǎng)絡(luò)傳輸?shù)暮臅r(shí):
圖中的各項(xiàng)標(biāo)注分別是(上面的時(shí)間線采用標(biāo)準(zhǔn)時(shí)間,與服務(wù)器和頁(yè)面的時(shí)間均無關(guān)):
start——頁(yè)面項(xiàng)服務(wù)器發(fā)起AJAX請(qǐng)求時(shí)的時(shí)間。
www_start——服務(wù)器響應(yīng)頁(yè)面的請(qǐng)求并返回時(shí)間戳給頁(yè)面的時(shí)間。
pc_start——頁(yè)面接受到服務(wù)器返回的時(shí)間戳并開始計(jì)時(shí)的時(shí)間。
www_end——服務(wù)器倒計(jì)時(shí)結(jié)束的時(shí)間。
pc_end——頁(yè)面倒計(jì)時(shí)結(jié)束的時(shí)間,同時(shí)也是用戶在倒計(jì)時(shí)結(jié)束那一刻點(diǎn)擊按鈕的時(shí)間。
end——服務(wù)器接收到用戶點(diǎn)擊信息的時(shí)間。
可以看出,即使在倒計(jì)時(shí)結(jié)束的那一刻(也就是秒殺開始那一刻)用戶就立即點(diǎn)擊鼠標(biāo),也會(huì)比實(shí)際開始搶拍的時(shí)間(www_end,即服務(wù)器倒計(jì)時(shí)結(jié)束的時(shí)間)晚一些(可以很容易的看出,這個(gè)時(shí)間差正好等于pc_start - start,也就是AJAX從開始發(fā)送到接收到響應(yīng)信息的耗時(shí))。如果有些內(nèi)行在頁(yè)面倒計(jì)時(shí)結(jié)束前用腳本發(fā)送請(qǐng)求,那么其他用戶可就虧大了。所以,我們要解決掉這個(gè)時(shí)間誤差的問題。
2.為了解決時(shí)間誤差的問題,我們將把頁(yè)面倒計(jì)時(shí)的時(shí)間縮短一小截(由上面的分析可以得出,這一小截正好等于pc_start - start),使得用戶在倒計(jì)時(shí)結(jié)束時(shí)發(fā)送給服務(wù)器的搶拍信息正好在服務(wù)器倒計(jì)時(shí)結(jié)束時(shí)被接收到:
圖中的各項(xiàng)標(biāo)注與Pic.1中相同(時(shí)間線采用標(biāo)準(zhǔn)時(shí)間,與服務(wù)器和頁(yè)面的時(shí)間均無關(guān)),新增的兩項(xiàng)標(biāo)注的含義如下:
old_pc_end——在未對(duì)網(wǎng)絡(luò)傳輸耗時(shí)進(jìn)行處理的情況下pc_end的時(shí)間。
old_end——在未對(duì)網(wǎng)絡(luò)傳輸耗時(shí)進(jìn)行處理的情況下end的時(shí)間。
由Pic.2可見,網(wǎng)絡(luò)傳輸耗時(shí)造成的時(shí)間誤差已經(jīng)完全被彌補(bǔ)了,彌補(bǔ)的方法是“將倒計(jì)時(shí)結(jié)束的時(shí)間提前pc_start - start”。但是解決了網(wǎng)絡(luò)傳輸耗時(shí)造成的誤差問題,還有用戶電腦時(shí)間和服務(wù)器時(shí)間不相同的問題,下面我們繼續(xù)討論。
3.用戶的電腦時(shí)間和服務(wù)器時(shí)間一定是有差異的,甚至差幾個(gè)時(shí)區(qū),怎么解決這個(gè)問題呢?方法的要點(diǎn)如下:
A. 當(dāng)頁(yè)面接收到服務(wù)器返回的時(shí)間戳www_t時(shí),立即開始計(jì)時(shí)。
B. 當(dāng)頁(yè)面接收到服務(wù)器返回的時(shí)間戳www_t時(shí),立即計(jì)算本地時(shí)間和服務(wù)器返回的時(shí)間戳的時(shí)間差t=new Date().getTime() - www_t*1000。
C. 仍然使用new Date().getTime()來計(jì)時(shí),而不是使用setInterval()函數(shù)(計(jì)時(shí)器很不穩(wěn)定,誤差也很大),但時(shí)間的顯示與程序的邏輯必須基于本地時(shí)間和上一步(B中)求得的時(shí)間偏差t。
結(jié)論要點(diǎn):
頁(yè)面從接收到服務(wù)器響應(yīng)的時(shí)間戳開始計(jì)時(shí),計(jì)時(shí)的時(shí)長(zhǎng)應(yīng)減掉AJAX從發(fā)送到接收整個(gè)過程的耗時(shí),計(jì)時(shí)過程則使用本地時(shí)間來實(shí)現(xiàn)(本地時(shí)間+時(shí)間偏差)。
有任何疑問或建議請(qǐng)留言,謝謝!
javascript小技巧:同步服務(wù)器時(shí)間、同步倒計(jì)時(shí)
之前在網(wǎng)上看到有人提問,如何在頁(yè)面上同步顯示服務(wù)器的時(shí)間,其實(shí)實(shí)現(xiàn)方法有幾種,可能一般人立馬就想到可以使用Ajax每隔一秒去請(qǐng)求服務(wù)器,然后將服務(wù)器獲取到時(shí)間顯示在頁(yè)面上,這樣雖然能夠?qū)崿F(xiàn),但存在一個(gè)很大的問題,那就是每隔一秒去請(qǐng)求服務(wù)器,這樣如果用戶多了,服務(wù)器就會(huì)崩潰(內(nèi)存占用率會(huì)很大),所以在我看來,這種方法不可行,我這里給出一種解決方案,能夠?qū)崿F(xiàn)同步服務(wù)器時(shí)間、同步倒計(jì)時(shí),卻不占用服務(wù)器太多資源,下面我給寫實(shí)現(xiàn)的思路
第一步,當(dāng)用戶第一次瀏覽頁(yè)面時(shí),服務(wù)器首先獲取當(dāng)前時(shí)間并顯示在頁(yè)面上(比如:顯示在ID為timebox span中)
第二步,設(shè)置一個(gè)每隔一秒就計(jì)算新的時(shí)間(新時(shí)間以服務(wù)器時(shí)間為初始值,然后每隔一秒累加一秒并生成新的時(shí)間)
第三步,顯示第二步計(jì)算的時(shí)間
是不是很簡(jiǎn)單,總結(jié)成一句話就是:以服務(wù)器時(shí)間為初始值,然后在頁(yè)面上自動(dòng)每隔一秒就累加一秒生成新時(shí)間,這樣就能保證與服務(wù)器時(shí)間同步了,誤差基本在幾秒內(nèi),應(yīng)該沒關(guān)系了,好了看一下實(shí)現(xiàn)的代碼吧:
<span id="timebox">11:21:55</span> //第一次將服務(wù)器時(shí)間顯示在這里 <script type="text/javascript"> $(function () { var oTime = $("#timebox"); var ts = oTime.text().split(":", 3); var tnums = [parseInt(ts[0]), parseInt(ts[1]), parseInt(ts[2])]; setInterval(function () { tnums = getNextTimeNumber(tnums[0], tnums[1], tnums[2]); showNewTime(tnums[0], tnums[1], tnums[2]); }, 1000); function showNewTime(h, m, s) { var timeStr = ("0" + h.toString()).substr(-2) + ":" + ("0" + m.toString()).substr(-2) + ":" + ("0" + s.toString()).substr(-2); oTime.text(timeStr); } function getNextTimeNumber(h, m, s) { if (++s == 60) { s = 0; } if (s == 0) { if (++m == 60) { m = 0; } } if (m == 0) { if (++h == 24) { h = 0; } } return [h, m, s]; } }); </script>
代碼很簡(jiǎn)單在此就不多作說明(我上面只顯示時(shí)分秒,大家也可以加上日期,加上日期可在當(dāng)h==0時(shí),直接從服務(wù)器獲取一個(gè)日期或完整的時(shí)間,作為一次時(shí)間的校對(duì)),不懂的可以在下面評(píng)論,我會(huì)及時(shí)回復(fù)的,然后按照這種思路來實(shí)現(xiàn)一下同步倒計(jì)時(shí),首先說明一下,什么是同步倒計(jì)時(shí),就是類似秒殺一樣,設(shè)置一個(gè)結(jié)束時(shí)間,然后計(jì)算當(dāng)前時(shí)間與結(jié)束時(shí)間之間間隔,而且必需保證在不同的電腦、瀏覽器上顯示的倒計(jì)時(shí)時(shí)間均相同,實(shí)現(xiàn)代碼如下:
<!DOCTYPE html> <html> <head> <title>同步倒計(jì)時(shí)</title> <script type="text/javascript" src="jquery-1.4.4.min.js"></script> </head> <body> <span id="timebox">1天00時(shí)00分12秒</span> <!--假設(shè):1天00時(shí)00分12秒是從服務(wù)器獲取的倒計(jì)時(shí)數(shù)據(jù)--> <script type="text/javascript"> $(function () { var tid = setInterval(function () { var oTimebox = $("#timebox"); var syTime = oTimebox.text(); var totalSec = getTotalSecond(syTime) - 1; if (totalSec >= 0) { oTimebox.text(getNewSyTime(totalSec)); } else { clearInterval(tid); } }, 1000); //根據(jù)剩余時(shí)間字符串計(jì)算出總秒數(shù) function getTotalSecond(timestr) { var reg = /\d+/g; var timenums = new Array(); while ((r = reg.exec(timestr)) != null) { timenums.push(parseInt(r)); } var second = 0, i = 0; if (timenums.length == 4) { second += timenums[0] * 24 * 3600; i = 1; } second += timenums[i] * 3600 + timenums[++i] * 60 + timenums[++i]; return second; } //根據(jù)剩余秒數(shù)生成時(shí)間格式 function getNewSyTime(sec) { var s = sec % 60; sec = (sec - s) / 60; //min var m = sec % 60; sec = (sec - m) / 60; //hour var h = sec % 24; var d = (sec - h) / 24;//day var syTimeStr = ""; if (d > 0) { syTimeStr += d.toString() + "天"; } syTimeStr += ("0" + h.toString()).substr(-2) + "時(shí)" + ("0" + m.toString()).substr(-2) + "分" + ("0" + s.toString()).substr(-2) + "秒"; return syTimeStr; } }); </script> </body> </html>
為了保證倒計(jì)時(shí)的精確度,我采用了先將倒計(jì)時(shí)時(shí)間間隔統(tǒng)一計(jì)算成秒,然后減1秒再重新生成時(shí)間格式,當(dāng)然也可以按照上面時(shí)間同步的例子,直接進(jìn)行時(shí)間減少,方法很多,我這個(gè)不一定是最優(yōu)的,歡迎大家交流,謝謝!
- JS 倒計(jì)時(shí)實(shí)現(xiàn)代碼(時(shí)、分,秒)
- JS實(shí)現(xiàn)倒計(jì)時(shí)(天數(shù)、時(shí)、分、秒)
- js代碼實(shí)現(xiàn)點(diǎn)擊按鈕出現(xiàn)60秒倒計(jì)時(shí)
- 簡(jiǎn)單易用的倒計(jì)時(shí)js代碼
- 2種簡(jiǎn)單的js倒計(jì)時(shí)方式
- 原生JS實(shí)現(xiàn)簡(jiǎn)單的倒計(jì)時(shí)功能示例
- js幾秒以后倒計(jì)時(shí)跳轉(zhuǎn)示例
- js實(shí)現(xiàn)點(diǎn)擊獲取驗(yàn)證碼倒計(jì)時(shí)效果
- 一個(gè)不錯(cuò)的js html頁(yè)面倒計(jì)時(shí)可精確到秒
- js實(shí)現(xiàn)酷炫倒計(jì)時(shí)動(dòng)畫
相關(guān)文章
js 為label標(biāo)簽和div標(biāo)簽賦值的方法
這篇文章介紹了js 為label標(biāo)簽和div標(biāo)簽賦值的方法,有需要的朋友可以參考一下2013-08-08用js實(shí)現(xiàn)控件的隱藏及style.visibility的使用
用js控制控件的隱藏,使用style.visibility實(shí)現(xiàn) ,具體代碼如下,感興趣的朋友可以參考下哈,希望對(duì)大家有所幫助2013-06-06JavaScript數(shù)組和對(duì)象的復(fù)制
本篇文章主要介紹了JavaScript數(shù)組和對(duì)象的復(fù)制的相關(guān)知識(shí)。具有很好的參考價(jià)值。下面跟著小編一起來看下吧2017-03-03js拖拉表格實(shí)現(xiàn)內(nèi)容計(jì)算
這篇文章主要為大家詳細(xì)介紹了js拖拉表格實(shí)現(xiàn)內(nèi)容計(jì)算,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04