說說JavaScript定時器

下面代碼的輸出結(jié)果:

for(var i = 0; i < 10; i++) {
    setTimeout(function() {
        console.log(i);
    }, 1000);
}

答案是:10個10

由此梦谜,我們來了解一下JavaScript的定時機制仔蝌。
眾所周知,JavaScript引擎是單線程的一喘,我們先來看一下JavaScript的線程圖示:


JavaScript線程圖示

如圖,一般瀏覽器內(nèi)核中至少常駐三個線程:JavaScript引擎線程嗜暴、GUI渲染線程凸克、瀏覽器事件觸發(fā)線程等。
其他線程產(chǎn)生的任務(wù)放到JavaScript引擎線程的隊列中闷沥,然后一個接一個的執(zhí)行萎战。

//代碼一
setTimeout(function() {
       console.log("123");
}, 1000);
//代碼二
setTimeout(function() {
       console.log("abc");
        setTimeout(arguments.callee, 1000);
}, 1000);
//代碼三
setInterval(function(){
        console.log("xyz");
}, 1000);

代碼一,延遲執(zhí)行狐赡,一秒后控制臺打印出123
代碼三撞鹉,循環(huán)執(zhí)行疟丙,每過一秒(<=1s)打印xyz
代碼二,也是循環(huán)執(zhí)行鸟雏,每過一秒(>=1s)打印abc
由于arguments.callee不推薦使用了享郊,所以代碼二可以等價的寫成:

setTimeout(function a() {
       console.log("abc");
        setTimeout(a, 1000);
}, 1000);

代碼二和代碼三是循環(huán)執(zhí)行,會一直執(zhí)行下去孝鹊,容易造成瀏覽器假死炊琉,沒反應(yīng),如果想要結(jié)束循環(huán)又活,可以通過如下方式:

//代碼二改成這樣苔咪,只執(zhí)行10次
var num=0;
setTimeout(function a() {
       num++;
       console.log("abc");
       if(num<10){
               setTimeout(a, 1000);
       }
}, 1000);
//代碼二也可以改成這樣,執(zhí)行11次
var num=0;
setTimeout(function a() {
       num++;
       console.log("abc");
       var timer=setTimeout(a, 1000);
       if(num>10){
               clearTimeout(timer);
       }
}, 1000);

同樣的道理柳骄,代碼三可以改成如下寫法:

//9次
var num=0;
setInterval(function(){
        num++;
        if(num<10){
               console.log("xyz");
        }
}, 1000);
//11次
var num=0;
var i=setInterval(function(){
        num++;
        console.log("xyz");
        if(num>10){
                clearInterval(i);
        }
}, 1000);

差不多就這樣吧团赏,總結(jié)一下:

  • JavaScript是單線程的
  • JavaScript引擎是基于事件驅(qū)動的
  • 定時器產(chǎn)生的異步事件會插入到JavaScript引擎的事件隊列中
  • JavaScript引擎執(zhí)行完同步事件后會去繼續(xù)執(zhí)行事件隊列中的異步事件,這就是為什么如下代碼先打印0再打印1的原因
setTimeout(function(){
    console.log(1);
},0);
console.log(0);
  • 兩個setTimeout構(gòu)成的循環(huán)可以使用clearTimeout()去結(jié)束循環(huán)
  • setInterval循環(huán)可以使用clearInterval()去結(jié)束循環(huán)

參考閱讀:
深入理解JavaScript定時機制
深入理解定時器系列第一篇——理解setTimeout和setInterval
瀏覽器UI多線程及對JavaScript單線程底層運行機制的理解

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末耐薯,一起剝皮案震驚了整個濱河市舔清,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌曲初,老刑警劉巖体谒,帶你破解...
    沈念sama閱讀 211,042評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異臼婆,居然都是意外死亡抒痒,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,996評論 2 384
  • 文/潘曉璐 我一進店門颁褂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來故响,“玉大人,你說我怎么就攤上這事痢虹”蝗ィ” “怎么了?”我有些...
    開封第一講書人閱讀 156,674評論 0 345
  • 文/不壞的土叔 我叫張陵奖唯,是天一觀的道長。 經(jīng)常有香客問我糜值,道長丰捷,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,340評論 1 283
  • 正文 為了忘掉前任寂汇,我火速辦了婚禮病往,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘骄瓣。我一直安慰自己停巷,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,404評論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著畔勤,像睡著了一般蕾各。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上庆揪,一...
    開封第一講書人閱讀 49,749評論 1 289
  • 那天式曲,我揣著相機與錄音,去河邊找鬼缸榛。 笑死吝羞,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的内颗。 我是一名探鬼主播钧排,決...
    沈念sama閱讀 38,902評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼均澳!你這毒婦竟也來了恨溜?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,662評論 0 266
  • 序言:老撾萬榮一對情侶失蹤负懦,失蹤者是張志新(化名)和其女友劉穎筒捺,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體纸厉,經(jīng)...
    沈念sama閱讀 44,110評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡系吭,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,451評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了颗品。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片肯尺。...
    茶點故事閱讀 38,577評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖躯枢,靈堂內(nèi)的尸體忽然破棺而出则吟,到底是詐尸還是另有隱情,我是刑警寧澤锄蹂,帶...
    沈念sama閱讀 34,258評論 4 328
  • 正文 年R本政府宣布氓仲,位于F島的核電站,受9級特大地震影響得糜,放射性物質(zhì)發(fā)生泄漏敬扛。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,848評論 3 312
  • 文/蒙蒙 一朝抖、第九天 我趴在偏房一處隱蔽的房頂上張望啥箭。 院中可真熱鬧,春花似錦治宣、人聲如沸急侥。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,726評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽坏怪。三九已至贝润,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間陕悬,已是汗流浹背题暖。 一陣腳步聲響...
    開封第一講書人閱讀 31,952評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留捉超,地道東北人胧卤。 一個月前我還...
    沈念sama閱讀 46,271評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像拼岳,于是被迫代替她去往敵國和親枝誊。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,452評論 2 348

推薦閱讀更多精彩內(nèi)容