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

首頁 > 編程 > JavaScript > 正文

理解javascript定時器中的setTimeout與setInterval

2019-11-20 10:31:52
字體:
供稿:網(wǎng)友

一、解釋說明

1、概述

setTimeout:在指定的延遲時間之后調(diào)用一個函數(shù)或者執(zhí)行一個代碼片段

setInterval:周期性地調(diào)用一個函數(shù)(function)或者執(zhí)行一段代碼。

2、語法

setTimeout:

var timeoutID = window.setTimeout(func, delay, [param1, param2, ...]);var timeoutID = window.setTimeout(code, delay);
  • timeoutID 是該延時操作的數(shù)字ID, 此ID隨后可以用來作為window.clearTimeout方法的參數(shù)
  • func 是你想要在delay毫秒之后執(zhí)行的函數(shù)
  • code 在第二種語法,是指你想要在delay毫秒之后執(zhí)行的代碼
  • delay 是延遲的毫秒數(shù) (一秒等于1000毫秒),函數(shù)的調(diào)用會在該延遲之后發(fā)生.但是實際的延遲時間可能會稍長一點
  • 標準瀏覽器與IE10支持第一種語法中向延遲函數(shù)傳遞額外參數(shù)的功能

setInterval

var intervalID = window.setInterval(func, delay[, param1, param2, ...]);var intervalID = window.setInterval(code, delay);
  • intervalID 是此重復操作的唯一辨識符,可以作為參數(shù)傳給clearInterval()。
  • func 是你想要重復調(diào)用的函數(shù)。
  • code 是另一種語法的應用,是指你想要重復執(zhí)行的一段字符串構(gòu)成的代碼
  • delay 是每次延遲的毫秒數(shù) (一秒等于1000毫秒),函數(shù)的每次調(diào)用會在該延遲之后發(fā)生。和setTimeout一樣,實際的延遲時間可能會稍長一點。
  • 標準瀏覽器與IE10支持第一種語法中向延遲函數(shù)傳遞額外參數(shù)的功能 
<script type="text/javascript">  setTimeout( function(param){ alert(param)} , 100, 'ok'); </script> 

簡單測試了下第五條,在我的電腦上面分別使用firefox與IE9測試,前者可以順利彈出ok,后者彈出了undefined。

 二、“this”問題

由setTimeout()調(diào)用的代碼運行在與所在函數(shù)完全分離的執(zhí)行環(huán)境上. 這會導致,這些代碼中包含的 this 關鍵字會指向 window (全局對象)對象,這和所期望的this的值是不一樣的。setInterval的情況類似。

<script type="text/javascript">  //this指向window  function shape(name) {    this.name = name;    this.timer = function(){alert('my shape is '+this.name)};    setTimeout(this.timer, 50);  }  new shape('rectangle');</script>

沒有被傳進去,分別用chrome,firefox和IE9實驗了下,都是這個結(jié)果。

解決方法一:

<script type="text/javascript">    function shape(name) {    this.name = name;    this.timer = function(){alert('my shape is '+this.name)};    var _this = this;    setTimeout(function() {_this.timer.call(_this)}, 50);  }  new shape('rectangle');</script>

設置一個局部變量_this,然后放到setTimeout的函數(shù)變量中,timer執(zhí)行call或apply,設置this值。

function能夠調(diào)用局部變量_this,多虧了Javascript的閉包。里面涉及了作用域鏈等知識,有興趣的可以自己去了解下,這里不展開了。

解決方法二:

這個方法有點高大上。自定義了setTimeout與setInterval。而且還擴展了低版本的IE瀏覽器,不支持向延遲函數(shù)傳遞額外參數(shù)的問題。

<script type="text/javascript">   //自定義setTimeout與setInterval  var __nativeST__ = window.setTimeout, __nativeSI__ = window.setInterval;   window.setTimeout = function (vCallback, nDelay /*, argumentToPass1, argumentToPass2, etc. */) {   var oThis = this, aArgs = Array.prototype.slice.call(arguments, 2);   return __nativeST__(vCallback instanceof Function ? function () {    vCallback.apply(oThis, aArgs);   } : vCallback, nDelay);  };     window.setInterval = function (vCallback, nDelay /*, argumentToPass1, argumentToPass2, etc. */) {   var oThis = this, aArgs = Array.prototype.slice.call(arguments, 2);   return __nativeSI__(vCallback instanceof Function ? function () {    vCallback.apply(oThis, aArgs);   } : vCallback, nDelay);  };     function shape(name) {    this.name = name;    this.timer = function(other){      alert('my shape is '+this.name);      alert('extra param is '+ other);    };  }  var rectangle = new shape('rectangle');  setTimeout.call(rectangle, rectangle.timer, 50, 'other');</script>

1、設置局部變量,賦值為原生的setTimeout與setInterval

2、擴展setTimeout與setInterval,aArgs通過分割arguments這個變量,獲取到額外的參數(shù)數(shù)組

3、用vCallback instanceof Function判斷這是不是一個函數(shù)或代碼,如果是函數(shù)就用apply執(zhí)行

4、setTimeout用call執(zhí)行,設定this對象,以及其它的func、delay等參數(shù)

5、順便擴展setTimeout,IE低版本的瀏覽器也能執(zhí)行額外參數(shù)

 三、setTimeout與setInterval之間的一個區(qū)別

<script type="text/javascript"> setTimeout(function(){  /* Some long block of code... */  setTimeout(arguments.callee, 100); }, 10);  setInterval(function(){  /* Some long block of code... */ }, 100);</script>

看上去,兩個功能是差不多的,但是里面其實是不一樣的。

setTimeout回調(diào)函數(shù)的執(zhí)行和上一次執(zhí)行之間的間隔至少有100ms(可能會更多,但不會少于100ms)

setInterval的回調(diào)函數(shù)將嘗試每隔100ms執(zhí)行一次,不論上次是否執(zhí)行完畢,時間間隔理論上是會<=delay的。

setInterval:

<script type="text/javascript">    function sleep(ms) {      var start = new Date();      while (new Date() - start <= ms) {}    }    var endTime = null;    var i = 0;        setInterval(count, 100);    function count() {      var elapsedTime = endTime ? (new Date() - endTime) : 100;      i++;      console.log('current count: ' + i + '.' + 'elapsed time: ' + elapsedTime + 'ms');      sleep(200);      endTime = new Date();    }</script>

從firefox的firebug可以查看到,時間間隔很不規(guī)則。

情況大致是這樣的:由于count函數(shù)的執(zhí)行時間遠大于setInterval的定時間隔,那么定時觸發(fā)線程就會源源不斷的產(chǎn)生異步定時事件,并放到任務隊列尾而不管它們是否已被處理,但一旦一個定時事件任務處理完,這些排列中的剩余定時事件就依次不間斷的被執(zhí)行。

setTimeout:

<script type="text/javascript">    function sleep(ms) {      var start = new Date();      while (new Date() - start <= ms) {}    }    var endTime = null;    var i = 0;    setTimeout(count, 100);    function count() {      var elapsedTime = endTime ? (new Date() - endTime) : 100;      i++;      console.log('current count: ' + i + '.' + 'elapsed time: ' + elapsedTime + 'ms');      sleep(200);      endTime = new Date();      setTimeout(count, 100);    }</script>  

以上就是本文的全部內(nèi)容,希望對大家學習javascript定時器有所幫助。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 新源县| 黔江区| 资溪县| 鸡西市| 榆林市| 砚山县| 墨玉县| 太原市| 江源县| 麻阳| 巴彦县| 张家口市| 张北县| 大厂| 雷波县| 永清县| 临澧县| 时尚| 徐闻县| 商都县| 汝州市| 临高县| 马山县| 九江市| 台南市| 洛宁县| 竹山县| 甘谷县| 瑞丽市| 商都县| 金秀| 宁远县| 房产| 新和县| 平潭县| 枞阳县| 津南区| 于都县| 大田县| 桦南县| 山东省|