• 详解promise、async和await的执行顺序


    1、题目和答案

    一道题题目:下面这段promise、async和await代码,请问控制台打印的顺序?

    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 66node v10中,正确输出是:

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

    2、知识点

    显然,这考察的是js中的事件循环和回调队列。注意以下几点:

    • Promise优先于setTimeout宏任务。所以,setTimeout回调会在最后执行。
    • Promise一旦被定义,就会立即执行。
    • Promiserejectresolve是异步执行的回调。所以,resolve()会被放到回调队列中,在主函数执行完和setTimeout前调用。
    • await执行完后,会让出线程。async标记的函数会返回一个Promise对象

    3、 难点

    最令人困惑的,就是async1 endpromise2之后输出

    在函数async1中,执行promise由于async2async标记的函数,所以默认返回promise对象)会发现resolve(),然后放入回调队列。

    接着执行下方的new Promise中的resolve()输出promise2,再回来输出async1 end

    其中,async1函数可以写成以下方式(便于理解):

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

    3、流程

    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. finish

    4、参考

    1. promise、async和await之执行顺序的那点事
    2. 半年工作经验今日头条和美团面试题面经分享
    3. 关于node和chrome运行结果不一样的详解

    欢迎技术交流,引用请注明出处。
    个人网站:godbmw.com
    Github:godbmw

  • 相关阅读:
    CSP-S2020总结
    题解-P6687 论如何玩转 Excel 表格
    题解-UVA12995 【Farey Sequence】
    题解-P4159 [SCOI2009] 【迷路】
    题解-SP2916【GSS5
    102. 二叉树的层序遍历
    力扣 160 相交链表 快慢指针 双指针
    3. 无重复字符的最长子串 滑动窗口
    最大连续1的个数 III
    B树和B+树
  • 原文地址:https://www.cnblogs.com/geyouneihan/p/9127324.html
Copyright © 2020-2023  润新知