setInterval計時器不準的問題解決方法(轉載)
setInterval計時器不準的問題解決方法
轉載 更新時間:2014年05月08日 15:45:32 作者: 我要評論
在js中如果打算使用setInterval進行倒數,計時等功能,往往是不準確的,針對這個問題,本文有個不錯的解決方案
在js中如果打算使用setInterval進行倒數,計時等功能,往往是不準確的,因為setInterval的回撥函式並不是到時後立即執行,而是等系統計算資源空閒下來後才會執行.而下一次觸發時間則是在setInterval回撥函式執行完畢之後才開始計時,所以如果setInterval內執行的計算過於耗時,或者有其他耗時任務在執行,setInterval的計時會越來越不準,延遲很厲害.
下面的程式碼可以說明這個問題
複製程式碼 程式碼如下:
var startTime = new Date().getTime();
var count = 0;
//耗時任務
setInterval(function(){
var i = 0;
while(i++ < 100000000);
}, 0);
setInterval(function(){
count++;
console.log(new Date().getTime() - (startTime + count * 1000));
}, 1000);
程式碼裡輸出了setInterval觸發時間和應該正確觸發時間的延遲毫秒數
複製程式碼 程式碼如下:
176
340
495
652
807
961
1114
1268
1425
1579
1734
1888
2048
2201
2357
2521
2679
2834
2996
......
可以看到延遲是越來越嚴重的.
為了在js裡可以使用相對準確的計時功能,我們可以
複製程式碼 程式碼如下:
var startTime = new Date().getTime();
var count = 0;
setInterval(function(){
var i = 0;
while(i++ < 100000000);
}, 0);
function fixed() {
count++;
var offset = new Date().getTime() - (startTime + count * 1000);
var nextTime = 1000 - offset;
if (nextTime < 0) nextTime = 0;
setTimeout(fixed, nextTime);
console.log(new Date().getTime() - (startTime + count * 1000));
}
setTimeout(fixed, 1000);
程式碼裡,通過1000(也就是週期時間)減去當前時間和準確時間的差距,來算出下次觸發的時間,從而修正了當前觸發的延遲.
下面是輸出
複製程式碼 程式碼如下:
186
200
230
271
158
899
900
899
900
899
899
899
902
899
418
202
232
266
145
174
192
214
242
268
149
179
214
......
可以看到雖然觸發時間並非絕對準確,但由於每次觸發都進行及時修正,所以並沒有造成誤差積累.