var period = 60 * 1000 * 60 * 2
var end
var date = new Date(end)
var interval = 1000
var loop = function () {
if (!end) { end = new Date().getTime() + period }
var diff = end - new Date().getTime()
var h = Math.floor(diff / (60 * 1000 * 60))
var hdiff = diff % (60 * 1000 * 60)
var m = Math.floor(hdiff / (60 * 1000))
var mdiff = hdiff % (60 * 1000)
var s = Math.floor(mdiff / (1000))
console.log(h, m, s)
setTimeout(loop, interval)
}
setTimeout(loop, interval)
上面是我寫的一個倒計時方法,在控制臺打印出的結(jié)果卻總是跳了一秒,如下所示:
不知這是為何?
可以參考我下面的代碼,其中countDown函數(shù)接收倒計時的起始時間和結(jié)束時間(ms)。
其基本思路時倒推結(jié)束時間之前的所有計時時刻,setTimeout調(diào)用時,其時間延遲根據(jù)當(dāng)前時刻與下一個計時時刻的時間差來設(shè)置,而不是一個固定值,這樣就能避免累計誤差。
參數(shù)可以比較靈活, 比如起始時間可以是函數(shù)調(diào)用的當(dāng)前時間之前,如countDown(Date.now()-5000, Date.now() + 2 * 3600 * 1000),這樣的話程序一開始會連續(xù)輸出已經(jīng)過去的所有計時時刻;如果起始時間在調(diào)用時間之后,那么會等到那個時間之后才開始倒計時,比如countDown(Date.now()+5000, Date.now() + 2 * 3600 * 1000)。
(function(){
var now = Date.now();
countDown(now, now + 2 * 3600 * 1000);
function countDown(start, end){
var interval = 1000,
nextTick = start + ((end - start) % interval);
var loopTime = function(){
var now = Date.now();
while(now >= nextTick){
showTime(end - nextTick);
nextTick += interval;
}
if(nextTick <= end){
setTimeout(arguments.callee, nextTick - now);
}
};
loopTime();
}
function showTime(ms){
var sTotal = Math.floor(ms/1000),
h = Math.floor(sTotal/3600),
m = Math.floor(sTotal%3600/60),
s = Math.floor(sTotal%60),
timeStr = ('0' + h).slice(-2) + ':' +
('0' + m).slice(-2) + ':' +
('0' + s).slice(-2);
console.log(timeStr);
}
})();解決了,第一次執(zhí)行方法的時候就已經(jīng)過了一秒了,所以在初始化end變量的時候就應(yīng)該減去一秒!
var period = 60 * 1000 * 60 * 2
var end
var date = new Date(end)
var interval = 1000
function loop() {
if (!end) { end = new Date().getTime() + period - interval / 1000 }
var diff = end - new Date().getTime()
var h = Math.floor(diff / (60 * 1000 * 60))
var hdiff = diff % (60 * 1000 * 60)
var m = Math.floor(hdiff / (60 * 1000))
var mdiff = hdiff % (60 * 1000)
var s = Math.floor(mdiff / (1000))
console.log(h, m, s)
setTimeout(loop, interval)
}
setTimeout(loop, interval)
上述代碼會造成執(zhí)行代碼的時間累加問題,為了解決這個問題,參考這篇文章,更精確的倒計時應(yīng)該是這樣的:
var period = 60 * 1000 * 60 * 2
var startTime = new Date().getTime();
var count = 0
var end = new Date().getTime() + period
var interval = 1000
var currentInterval = interval
function loop() {
count++
var offset = new Date().getTime() - (startTime + count * interval); // 代碼執(zhí)行所消耗的時間
var diff = end - new Date().getTime()
var h = Math.floor(diff / (60 * 1000 * 60))
var hdiff = diff % (60 * 1000 * 60)
var m = Math.floor(hdiff / (60 * 1000))
var mdiff = hdiff % (60 * 1000)
var s = mdiff / (1000)
var sCeil = Math.ceil(s)
var sFloor = Math.floor(s)
currentInterval = interval - offset // 得到下一次循環(huán)所消耗的時間
console.log('時:'+h, '分:'+m, '毫秒:'+s, '秒向上取整:'+sCeil, '代碼執(zhí)行時間:'+offset, '下次循環(huán)間隔'+currentInterval) // 打印 時 分 秒 代碼執(zhí)行時間 下次循環(huán)間隔
setTimeout(loop, currentInterval)
}
setTimeout(loop, currentInterval)
感謝各位不吝賜教。
這個需求其實用setInterval更好,還不會出現(xiàn)你上述的問題:
var period = 60 * 1000 * 60 * 2
var end
var date = new Date(end)
var interval = 1000
var loop = function () {
if (!end) { end = new Date().getTime() + period }
var diff = end - new Date().getTime()
var h = Math.floor(diff / (60 * 1000 * 60))
var hdiff = diff % (60 * 1000 * 60)
var m = Math.floor(hdiff / (60 * 1000))
var mdiff = hdiff % (60 * 1000)
var s = Math.floor(mdiff / (1000))
console.log(h, m, s)
}
setInterval(loop, interval)var period = 60 * 1000 * 60 * 2
var end
var date = new Date(end)
var interval = 100
var diff,h,hdiff,m,mdiff,s, res, res_tmp;
var loop = function () {
if (!end) { end = new Date().getTime() + period }
diff = end - new Date().getTime()
h = Math.floor(diff / (60 * 1000 * 60))
hdiff = diff % (60 * 1000 * 60)
m = Math.floor(hdiff / (60 * 1000))
mdiff = hdiff % (60 * 1000)
s = Math.floor(mdiff / (1000))
res_tmp = [h, m, s].join('-');
if (!res || res != res_tmp) {
res = res_tmp;
console.log(res)
}
setTimeout(loop, interval)
}
setTimeout(loop, interval)
降低時間間隔。來排除setTimeout帶來的偏差累計
北大青鳥APTECH成立于1999年。依托北京大學(xué)優(yōu)質(zhì)雄厚的教育資源和背景,秉承“教育改變生活”的發(fā)展理念,致力于培養(yǎng)中國IT技能型緊缺人才,是大數(shù)據(jù)專業(yè)的國家
達(dá)內(nèi)教育集團(tuán)成立于2002年,是一家由留學(xué)海歸創(chuàng)辦的高端職業(yè)教育培訓(xùn)機(jī)構(gòu),是中國一站式人才培養(yǎng)平臺、一站式人才輸送平臺。2014年4月3日在美國成功上市,融資1
北大課工場是北京大學(xué)校辦產(chǎn)業(yè)為響應(yīng)國家深化產(chǎn)教融合/校企合作的政策,積極推進(jìn)“中國制造2025”,實現(xiàn)中華民族偉大復(fù)興的升級產(chǎn)業(yè)鏈。利用北京大學(xué)優(yōu)質(zhì)教育資源及背
博為峰,中國職業(yè)人才培訓(xùn)領(lǐng)域的先行者
曾工作于聯(lián)想擔(dān)任系統(tǒng)開發(fā)工程師,曾在博彥科技股份有限公司擔(dān)任項目經(jīng)理從事移動互聯(lián)網(wǎng)管理及研發(fā)工作,曾創(chuàng)辦藍(lán)懿科技有限責(zé)任公司從事總經(jīng)理職務(wù)負(fù)責(zé)iOS教學(xué)及管理工作。
浪潮集團(tuán)項目經(jīng)理。精通Java與.NET 技術(shù), 熟練的跨平臺面向?qū)ο箝_發(fā)經(jīng)驗,技術(shù)功底深厚。 授課風(fēng)格 授課風(fēng)格清新自然、條理清晰、主次分明、重點難點突出、引人入勝。
精通HTML5和CSS3;Javascript及主流js庫,具有快速界面開發(fā)的能力,對瀏覽器兼容性、前端性能優(yōu)化等有深入理解。精通網(wǎng)頁制作和網(wǎng)頁游戲開發(fā)。
具有10 年的Java 企業(yè)應(yīng)用開發(fā)經(jīng)驗。曾經(jīng)歷任德國Software AG 技術(shù)顧問,美國Dachieve 系統(tǒng)架構(gòu)師,美國AngelEngineers Inc. 系統(tǒng)架構(gòu)師。