浏览器定时器问题

时间:2021-04-29 18:11:08

  最近在做一个h5小游戏,游戏里有这样一个场景,两个倒计时同时在一分钟的时候开始倒计时,两者几乎是同时开始的,理论上,我希望的结果是他们能够几乎同时倒计时到0,但是,实际发现,在一段时间内,他们产生了很大的误差(大于一秒),于是,真正认识到js的定时器是那么的不靠谱了。为啥不靠谱。本篇文章来理理总结下。

  浏览器中的主要的定时器有两种,世人皆知:setTimeout和setInterval,这两个方法让程序在浏览器中的延时执行,为什么说是浏览器中而不是js中,因为定时器方法是浏览器宿主提供的,而非js本身自带,脱离了浏览器也许就不管用了。
使用实例
1 var timerId=setTimeout(function(){
2      alert("我要延时两秒来执行");
3 },2000);
1 var timerId1= setInterval(function(){
2    alert("我一秒钟执行一次");
3 },1000);

如果一切正常,第一段代码将在两秒后弹出弹窗,第二段代码将没一秒钟弹出一次弹窗。一切总是那么的顺畅。我们仿佛看到一个掐着秒表的调度员在调度和控制这些。

再看一段代码
1 setTimeout(function(){
2    alert(1);
3 },1000);
4 alert(2);

执行这段代码,我们发现先弹出了2,再弹出了1;也就是说我们发现定时器的一个特性,那就是“异步执行”!

  当然js是单线程的,为何能产生了异步的特性。回想下,在我们的js代码里,还有谁是异步呢,很自然,我们想到了ajax的异步。还有鼠标点击事件的异步,
这是为什么呢?难道还有其他线程在操控着这些异步。答案是肯定的。
 
回顾下几个知识点:
 
1.浏览器5个常驻线程
    js引擎分配的线程
    GUI渲染页面的线程
    浏览器事件线程
    浏览器定时器触发线程
    浏览器http请求线程
 
那么就很好理解了,js引擎的线程按照他的单线程按部就班的执行着代码。GUI负责渲染页面,但是他和js引擎的线程是互斥的,也就是只允许同时又一个线程在执行。浏览器事件线程负责处理浏览器点击等事件,浏览器定时器线程负责处理定时器的任务安排,http请求线程负责处理网络请求。
 
一切看上去很井然有序,那既然有专门的线程去安排这些事情,为何文章开篇会产生定时误差呢,
  原来浏览器的多线程其实只是处理事件的触发,至于事件的回调的执行,还是交给js的引擎线程来执行的,因为js引擎是单线程的,所以在js引擎中维护着一个队列,当这些事件触发的时候,就会把回调插入到这个位于正常js代码的队列中,队列是在正常代码块的末尾位置。正因为最终他们的回调还是单线程执行的,所以一旦上一步操作中有比较耗时间的操作,定时操作就会向后产生延误。那么就会出现上面开题所说的情况,而且这种延误会产生叠加。看下下面这张图,再理解下,为何上面的代码块先执行了alert(2)然后再执行了alert(1);
浏览器定时器问题图片来源(http://hao.jser.com/archive/8414/)

 

这下似乎问题都出来了,在我的小游戏当中,维持着很多缓动定时器还有一些轮询的ajax,所以可以理解有很多耗时的任务,他们阻碍了定时器的按部就班。
  那如何修复这种情况呢,毕竟问题还是要解决的。
 
  两种解决方案
  1.尽量使用setTimeout来替代setInterval,具体做法是:
   
1 function timer(fn,delay){
2      var s=setTimeout(function(){
3           fn();
4           timer(fn,delay);
5     },delay)      
6 }    

这种做法的原理是,每次执行setTimeout的时候都会重新触发定时器事件,重新插入队列事件,延误会再下次被清零,避免了连续的队列延迟产生叠加

2.即时校准

 1 var count=0;
 2 var t1=(new date()).getTime();
 3 var delay=1000;
 4 function timer(fn,delay){
 5      var s=setTimeout(function(){
 6         count++;
 7         fn();
 8         var t2=(new date()).getTime();
 9         var newdelay = delayer - (t2-t1-count*delay);
10         timer(fn,newdelay);
11     })
12 }    
13 timer(function(){
14     console.log(111);
15 },delay);

这种方式比起第一种方式更加暴力,效果明显,保证了时间最终的基本一致,但是会有因为校准而产生的时间跳跃现象。

好,问题算是解决了。

定时器还会涉及一些问题
1.那就是定时器的作用域里 this指向的肯定是window 除非你强制改变。
2.定时器哪怕你delay设置为0,也是会被插入到最后的队列中执行的。
3.定时器执行后会返回一个定时的索引给你,你可以通过clearTimeout 和clearInterval来定点清除
 
 
就先总结到这里了!