国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > JavaScript > 正文

Javascript實(shí)現(xiàn)商品秒殺倒計(jì)時(shí)(時(shí)間與服務(wù)器時(shí)間同步)

2019-11-20 11:33:34
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

現(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è)置等問(wèn)題)。

2.要考慮網(wǎng)絡(luò)傳輸?shù)暮臅r(shí)。

3.獲取時(shí)間時(shí)可直接從AJAX的響應(yīng)頭中讀取(通過(guò)getResponseHeader('Date')來(lái)獲得),服務(wù)器端不需要專(zhuān)門(mén)寫(xiě)時(shí)間生成腳本。

過(guò)程分析:

1.從服務(wù)器讀到一個(gè)時(shí)間戳之后便開(kāi)始計(jì)時(shí),不考慮網(wǎng)絡(luò)傳輸?shù)暮臅r(shí):


圖中的各項(xiàng)標(biāo)注分別是(上面的時(shí)間線采用標(biāo)準(zhǔn)時(shí)間,與服務(wù)器和頁(yè)面的時(shí)間均無(wú)關(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í)間戳并開(kāi)始計(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é)束的那一刻(也就是秒殺開(kāi)始那一刻)用戶就立即點(diǎn)擊鼠標(biāo),也會(huì)比實(shí)際開(kāi)始搶拍的時(shí)間(www_end,即服務(wù)器倒計(jì)時(shí)結(jié)束的時(shí)間)晚一些(可以很容易的看出,這個(gè)時(shí)間差正好等于pc_start - start,也就是AJAX從開(kāi)始發(fā)送到接收到響應(yīng)信息的耗時(shí))。如果有些內(nèi)行在頁(yè)面倒計(jì)時(shí)結(jié)束前用腳本發(fā)送請(qǐng)求,那么其他用戶可就虧大了。所以,我們要解決掉這個(gè)時(shí)間誤差的問(wèn)題。

2.為了解決時(shí)間誤差的問(wèn)題,我們將把頁(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í)間均無(wú)關(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可見(jiàn),網(wǎng)絡(luò)傳輸耗時(shí)造成的時(shí)間誤差已經(jīng)完全被彌補(bǔ)了,彌補(bǔ)的方法是“將倒計(jì)時(shí)結(jié)束的時(shí)間提前pc_start - start”。但是解決了網(wǎng)絡(luò)傳輸耗時(shí)造成的誤差問(wèn)題,還有用戶電腦時(shí)間和服務(wù)器時(shí)間不相同的問(wèn)題,下面我們繼續(xù)討論。

3.用戶的電腦時(shí)間和服務(wù)器時(shí)間一定是有差異的,甚至差幾個(gè)時(shí)區(qū),怎么解決這個(gè)問(wèn)題呢?方法的要點(diǎn)如下:

A. 當(dāng)頁(yè)面接收到服務(wù)器返回的時(shí)間戳www_t時(shí),立即開(kāi)始計(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()來(lái)計(jì)時(shí),而不是使用setInterval()函數(shù)(計(jì)時(shí)器很不穩(wěn)定,誤差也很大),但時(shí)間的顯示與程序的邏輯必須基于本地時(shí)間和上一步(B中)求得的時(shí)間偏差t。

結(jié)論要點(diǎn):

頁(yè)面從接收到服務(wù)器響應(yīng)的時(shí)間戳開(kāi)始計(jì)時(shí),計(jì)時(shí)的時(shí)長(zhǎng)應(yīng)減掉AJAX從發(fā)送到接收整個(gè)過(guò)程的耗時(shí),計(jì)時(shí)過(guò)程則使用本地時(shí)間來(lái)實(shí)現(xiàn)(本地時(shí)間+時(shí)間偏差)。

有任何疑問(wèn)或建議請(qǐng)留言,謝謝!

javascript小技巧:同步服務(wù)器時(shí)間、同步倒計(jì)時(shí)

之前在網(wǎng)上看到有人提問(wèn),如何在頁(yè)面上同步顯示服務(wù)器的時(shí)間,其實(shí)實(shí)現(xiàn)方法有幾種,可能一般人立馬就想到可以使用Ajax每隔一秒去請(qǐng)求服務(wù)器,然后將服務(wù)器獲取到時(shí)間顯示在頁(yè)面上,這樣雖然能夠?qū)崿F(xiàn),但存在一個(gè)很大的問(wèn)題,那就是每隔一秒去請(qǐng)求服務(wù)器,這樣如果用戶多了,服務(wù)器就會(huì)崩潰(內(nèi)存占用率會(huì)很大),所以在我看來(lái),這種方法不可行,我這里給出一種解決方案,能夠?qū)崿F(xiàn)同步服務(wù)器時(shí)間、同步倒計(jì)時(shí),卻不占用服務(wù)器太多資源,下面我給寫(xiě)實(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)該沒(méi)關(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)單在此就不多作說(shuō)明(我上面只顯示時(shí)分秒,大家也可以加上日期,加上日期可在當(dāng)h==0時(shí),直接從服務(wù)器獲取一個(gè)日期或完整的時(shí)間,作為一次時(shí)間的校對(duì)),不懂的可以在下面評(píng)論,我會(huì)及時(shí)回復(fù)的,然后按照這種思路來(lái)實(shí)現(xiàn)一下同步倒計(jì)時(shí),首先說(shuō)明一下,什么是同步倒計(jì)時(shí),就是類(lèi)似秒殺一樣,設(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)的,歡迎大家交流,謝謝!

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 东台市| 商丘市| 深泽县| 随州市| 张北县| 兴文县| 武宣县| 漯河市| 长兴县| 紫云| 嘉善县| 湖州市| 额济纳旗| 兴山县| 封丘县| 寿宁县| 云阳县| 晴隆县| 财经| 伊宁市| 拉萨市| 文登市| 土默特右旗| 梅州市| 扶余县| 益阳市| 宜宾市| 安新县| 灵宝市| 洛阳市| 蕉岭县| 和龙市| 樟树市| 莲花县| 武安市| 阿拉尔市| 那曲县| 扶风县| 盘锦市| 云梦县| 西畴县|