JS異步之宏隊列與微隊列


1. 原理圖

 

 2. 說明

JS 中用來存儲待執行回調函數的隊列包含 2 個不同特定的列隊

  • 宏列隊:用來保存待執行的宏任務(回調),比如:定時器回調、DOM 事件回調、ajax 回調

  • 微列隊:用來保存待執行的微任務(回調),比如:promise的回調、MutationObserver 的回調

JS 執行時會區別這 2 個隊列

  • JS 引擎首先必須先執行所有的初始化同步任務代碼

  • 每次准備取出第一個宏任務執行前, 都要將所有的微任務一個一個取出來執行,也就是優先級比宏任務高,且與微任務所處的代碼位置無關

下面這個例子可以看出 Promise 要先於 setTimeout 執行:

  setTimeout(() => { //立即放入宏隊列
      console.log('timeout callback1()')
      Promise.resolve(3).then(
        value => { //立即放入微隊列
          console.log('Promise onResolved3()', value)
        }
      )
    }, 0)

    setTimeout(() => { //立即放入宏隊列
      console.log('timeout callback2()')
    }, 0)

    Promise.resolve(1).then(
      value => { //立即放入微隊列
        console.log('Promise onResolved1()', value)
        setTimeout(() => {
          console.log('timeout callback3()', value)
        }, 0)
      }
    )

    Promise.resolve(2).then(
      value => { //立即放入微隊列
        console.log('Promise onResolved2()', value)
      }
    )

    // Promise onResolved1() 1
    // Promise onResolved2() 2
    // timeout callback1()
    // Promise onResolved3() 3
    // timeout callback2()
    // timeout callback3() 1

3. 相關面試題

代碼一:

  // 3 7 4 1 2 5
    /*
    宏: []
    微: []
    */
    const first = () => (new Promise((resolve, reject) => {
      console.log(3)
      let p = new Promise((resolve, reject) => {
        console.log(7)
        setTimeout(() => {
          console.log(5)
          resolve(6) //會被忽略,因為會先執行微隊列里的resolve(1),此時狀態已經改變過了,且狀態只能改變一次
        }, 0)
        resolve(1)
      })
      resolve(2)
      p.then((arg) => {
        console.log(arg)
      })
    }))

    first().then((arg) => {
      console.log(arg)
    })
    console.log(4)

代碼二:

  // 1 7 2 3 8 4 6 5 0     
    setTimeout(() => {
      console.log("0")
    }, 0)
    new Promise((resolve, reject) => {
      console.log("1")
      resolve()
    }).then(() => {
      console.log("2")
      new Promise((resolve, reject) => {
        console.log("3")
        resolve()
      }).then(() => {
        console.log("4")
      }).then(() => {
        console.log("5")
      })
    }).then(() => {
      console.log("6")
    })

    new Promise((resolve, reject) => {
      console.log("7")
      resolve()
    }).then(() => {
      console.log("8")
    })


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM