美文网首页
事件循环和回调队列的实战理解

事件循环和回调队列的实战理解

作者: _章鱼小丸子 | 来源:发表于2019-05-23 15:51 被阅读0次

通过一道常见的题目,解释事件循环和回调队列机制

async function async1(){
  console.log('async1 start')
  await async2()
  console.log('async1 end')
}
async function async2(){
  console.log('async2')
}
console.log('script start')
setTimeout(function(){
  console.log('setTimeout') 
},0)  
async1();
new Promise(function(resolve){
  console.log('promise1')
  resolve();
}).then(function(){
  console.log('promise2')
})
console.log('script end')

在Chrome 66和node v10中,此题的正确输出是:

script start
async1 start
async2
promise1
script end
promise2
async1 end
setTimeout

知识点

  • 宏任务
    一般包括包括:整体代码script,setTimeout,setInterval,setImmediate,MessageChannel
  • 微任务
    一般包括:Promise,process.nextTick
  • 执行顺序
    首先在 宏任务的队列(这个队列也被叫做 task queue)中取出第一个任务,执行完毕后取出微任务队列中的所有任务顺序执行;之后再取一个宏任务,周而复始,直至两个队列的任务都取完。
  • 优先级
     1. 宏任务的优先级:主代码块 > setImmediate > MessageChannel > setTimeout / setInterval
     2. 微任务的优先级:process.nextTick > Promise > MutationObserver
  • Promise
     1. Promise一旦被定义,就会立即执行。
     2. Promise的reject和resolve是异步执行的回调。所以,resolve()会被放到回调队列中
  • async/await
     1. await执行完后,会让出线程。
     2. async标记的函数会返回一个Promise对象

难点

为了便于理解,async1函数可以理解为以下方式:

async function async1(){
  console.log('async1 start')
  async2().then( () => {
    console.log( 'async1 end ')
  })
}

流程

  1. console.log('script start')输出:script start
  2. setTimeout被放在最后调用
  3. 执行async1函数,输出async1 start。然后,进入async2函数,输出async2,并返回Promise对象。回到async1,由于await,让出线程,async2函数返回的Promise放在回调队列。
  4. 新new了一个Promise对象,输出promise1。其中的resolve()被放在回调队列。
  5. console.log('script end')输出:script end
  6. 执行回调队列中,async1返回的Promise对象,对象产生的resolve被放入对调队列。这里不输出任何值。
  7. 执行回调队列中,下方Promise显式声明的resolve,输出promise2。
  8. 执行回调队列中,由于async1函数返回的promise对象的resolve,输出async1 end。
  9. 执行回调队列中,最后的setTimeout,输出setTimeout
  10. 完成

相关文章

  • 事件循环和回调队列的实战理解

    通过一道常见的题目,解释事件循环和回调队列机制 在Chrome 66和node v10中,此题的正确输出是: 知识...

  • 3,promise async/await settimeo

    这题主要考察的是这三者在事件循环中的区别,事件循环中分为宏任务队列和微任务队列。 其中setTimeout的回调函...

  • JS中的Event Loop

    事件和回调函数 任务队列其实是事件的一个队列,也可以理解为消息队列,当IO设备完成一个任务的时候,就会在任务队列中...

  • 中高级前端面试题

    事件循环机制 同步:执行栈 异步:任务队列 1.宏任务 点击回调 settimeout 2.微任务 当前task执...

  • 事件循环

    事件触发不会马上执行回调,会加入队列,队列中按照先进先出的顺序,逐个执行事件绑定的回调方法 新事件产生后会插在队尾...

  • JS调用栈/回调队列/事件循环

    调用栈 JS是单线程,一次只能做一件事执行一个函数即入栈,函数return后即出栈 阻塞/异步/回调队列/事件循环...

  • JS 运行机制

    关于Js的运行机制,经常会看到一下重要概念:单线程,事件,回调,事件循环,执行栈,任务队列,异步以及相关的函数se...

  • JS 异步之宏队列 与 微队列

    # 前言 JS 中用来存储待执行回调函数的队列包含 宏队列 和 微队列 宏队列:用来保存待执行的宏任务(回调),比...

  • Javascript异步解决方案的发展历程

    1.回调函数 优点:便于理解缺点:回调地狱,不能捕获错误 2.事件监听 容易理解,可以绑定多个事件,每个事件可以指...

  • 宏任务((macro)task)和微任务(microtask)

    宏任务 (macro)task,可以理解是每次执行栈执行的代码就是一个宏任务(包括每次从事件队列中获取一个事件回调...

网友评论

      本文标题:事件循环和回调队列的实战理解

      本文链接:https://www.haomeiwen.com/subject/opfbzqtx.html