JS中的Event Loop和異步執(zhí)行

Event Loop

JS是一門單線程的非阻塞的腳本語言没隘,只有一個(gè)主線程來處理所有任務(wù)届宠。當(dāng)JS執(zhí)行一系列任務(wù)時(shí)蛤袒,由于JS是單線程的,同一時(shí)間只能處理一個(gè)任務(wù)塌衰,于是這些任務(wù)就在執(zhí)行棧中排隊(duì)诉稍,JS會(huì)依次執(zhí)行這些任務(wù)。當(dāng)JS執(zhí)行一項(xiàng)異步任務(wù)(如I/O)時(shí)杯巨,主線程不會(huì)一直等待其返回結(jié)果,而是掛起(pending)這個(gè)任務(wù)努酸,繼續(xù)執(zhí)行執(zhí)行棧中的其他任務(wù)服爷,等異步任務(wù)返回結(jié)果時(shí)再根據(jù)一定規(guī)則去執(zhí)行相應(yīng)的回調(diào)。當(dāng)異步任務(wù)返回結(jié)果時(shí)获诈,JS會(huì)將這個(gè)異步任務(wù)加入與當(dāng)前執(zhí)行棧不同的另一個(gè)隊(duì)列--事件隊(duì)列仍源。被加入事件隊(duì)列的異步任務(wù)的回調(diào)不會(huì)立即被執(zhí)行,而是等待當(dāng)前執(zhí)行棧中的任務(wù)執(zhí)行完畢舔涎,主線程閑置時(shí)笼踩,JS從事件隊(duì)列中,取出排在第一位的任務(wù)亡嫌,將對(duì)應(yīng)回調(diào)放入執(zhí)行棧嚎于,并執(zhí)行其中的同步代碼掘而,如此反復(fù),形成一個(gè)無限循環(huán)于购,稱為事件循環(huán)(Event Loop)袍睡。

Macro Task 和 Micro Task

異步任務(wù)并不相同,它們的執(zhí)行順序也有差異价涝。不同的異步任務(wù)分為兩類:宏任務(wù)(Macro Task)和微任務(wù)(Micro Task)女蜈。
以下事件屬于宏任務(wù):

  • setTimeout()
  • setInterval()

以下事件屬于微任務(wù):

  • Promise.then()

當(dāng)前執(zhí)行棧中任務(wù)執(zhí)行完畢時(shí),JS會(huì)查看微任務(wù)隊(duì)列中是否有事件色瘩,如果沒有,會(huì)去宏任務(wù)隊(duì)列中取出排在第一位的事件并把其對(duì)應(yīng)回調(diào)放入當(dāng)前執(zhí)行棧執(zhí)行逸寓;如果微任務(wù)隊(duì)列中有事件居兆,會(huì)依次執(zhí)行事件對(duì)應(yīng)的回調(diào),直到微任務(wù)隊(duì)列清空竹伸,再去宏任務(wù)隊(duì)列中取出排在第一位的事件并把其對(duì)應(yīng)回調(diào)放入當(dāng)前執(zhí)行棧執(zhí)行泥栖,如此循環(huán)。即同步代碼執(zhí)行完畢之后勋篓,先清空微任務(wù)吧享,再執(zhí)行第一個(gè)宏任務(wù),同一次事件循環(huán)中譬嚣,微任務(wù)永遠(yuǎn)在宏任務(wù)之前執(zhí)行钢颂。
這樣就能解釋以下代碼:

      console.log('script start')

      let timer1 = setTimeout(() => {
        console.log('timer1')
        let promise1 = new Promise((resolve, reject) => {
          console.log('timer1-Promise1')
          resolve()
        })

        promise1.then(() => {
          console.log('timer1-Promise1-then')
        })
      }, 0)

      let timer2 = setTimeout(() => {
        console.log('timer2')
      }, 0)

      let promise2 = new Promise((resolve, reject) => {
        console.log('Promise2')
        resolve()
      })

      promise2
        .then(() => {
          console.log('Promise2-then1')
        })
        .then(() => {
          console.log('Promise2-then2')
        })

      console.log('script end')

執(zhí)行結(jié)果:


image.png

簡單分析:
主線程依次執(zhí)行這段代碼,同步代碼直接執(zhí)行拜银,異步代碼掛起殊鞭,加入事件隊(duì)列。
以下代碼直接執(zhí)行:

      console.log('script start')

      let promise2 = new Promise((resolve, reject) => {
        console.log('Promise2')
        resolve()
      })

      console.log('script end')

注意:作為Promise參數(shù)的這個(gè)函數(shù)是同步代碼尼桶,會(huì)直接執(zhí)行操灿。因此前三個(gè)打印結(jié)果依次是
script start
Promise2
script end

以下事件被加入宏任務(wù):

timer1

timer2

以下事件被加入微任務(wù):

      promise2
        .then()
        .then()

同步代碼執(zhí)行完畢,執(zhí)行棧清空泵督,主線程查看微任務(wù)隊(duì)列趾盐,執(zhí)行相應(yīng)回調(diào)清空微任務(wù)隊(duì)列,打印出
Promise2-then1
Promise2-then2

微任務(wù)隊(duì)列清空完畢后小腊,主線程查看宏任務(wù)隊(duì)列救鲤,執(zhí)行排在第一位的事件(timer1)對(duì)應(yīng)回調(diào)。
timer1的回調(diào)又開始了新的Event Loop
同步代碼:

        console.log('timer1')
        let promise1 = new Promise((resolve, reject) => {
          console.log('timer1-Promise1')
          resolve()
        })

直接執(zhí)行溢豆,打印出:
timer1
timer1-Promise1

以下事件加入微任務(wù):

promise1.then()

同步代碼執(zhí)行完畢蜒简,清空微任務(wù)隊(duì)列,打印出:
timer1-Promise1-then
此時(shí)宏任務(wù)隊(duì)列排在第一位的是timer2漩仙,主線程取出timer2執(zhí)行相應(yīng)回調(diào)搓茬,打印出:
timer2
整個(gè)腳本執(zhí)行完畢犹赖。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市卷仑,隨后出現(xiàn)的幾起案子峻村,更是在濱河造成了極大的恐慌,老刑警劉巖锡凝,帶你破解...
    沈念sama閱讀 221,576評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件粘昨,死亡現(xiàn)場離奇詭異,居然都是意外死亡窜锯,警方通過查閱死者的電腦和手機(jī)张肾,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,515評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來锚扎,“玉大人吞瞪,你說我怎么就攤上這事〖菘祝” “怎么了芍秆?”我有些...
    開封第一講書人閱讀 168,017評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長翠勉。 經(jīng)常有香客問我妖啥,道長,這世上最難降的妖魔是什么对碌? 我笑而不...
    開封第一講書人閱讀 59,626評(píng)論 1 296
  • 正文 為了忘掉前任荆虱,我火速辦了婚禮,結(jié)果婚禮上俭缓,老公的妹妹穿的比我還像新娘克伊。我一直安慰自己,他們只是感情好华坦,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,625評(píng)論 6 397
  • 文/花漫 我一把揭開白布愿吹。 她就那樣靜靜地躺著,像睡著了一般惜姐。 火紅的嫁衣襯著肌膚如雪犁跪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,255評(píng)論 1 308
  • 那天歹袁,我揣著相機(jī)與錄音坷衍,去河邊找鬼。 笑死条舔,一個(gè)胖子當(dāng)著我的面吹牛枫耳,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播孟抗,決...
    沈念sama閱讀 40,825評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼迁杨,長吁一口氣:“原來是場噩夢啊……” “哼钻心!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起铅协,我...
    開封第一講書人閱讀 39,729評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤捷沸,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后狐史,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體痒给,經(jīng)...
    沈念sama閱讀 46,271評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,363評(píng)論 3 340
  • 正文 我和宋清朗相戀三年骏全,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了苍柏。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,498評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡姜贡,死狀恐怖序仙,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情鲁豪,我是刑警寧澤,帶...
    沈念sama閱讀 36,183評(píng)論 5 350
  • 正文 年R本政府宣布律秃,位于F島的核電站爬橡,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏棒动。R本人自食惡果不足惜糙申,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,867評(píng)論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望船惨。 院中可真熱鬧柜裸,春花似錦、人聲如沸粱锐。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,338評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽怜浅。三九已至铐然,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間恶座,已是汗流浹背搀暑。 一陣腳步聲響...
    開封第一講書人閱讀 33,458評(píng)論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留跨琳,地道東北人自点。 一個(gè)月前我還...
    沈念sama閱讀 48,906評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長得像脉让,于是被迫代替她去往敵國和親桂敛。 傳聞我的和親對(duì)象是個(gè)殘疾皇子功炮,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,507評(píng)論 2 359

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