zoukankan      html  css  js  c++  java
  • 详解promise、async和await的执行顺序

    前言

    对于promise、async和await的执行顺序,很多人都容易弄混,也有很多人只愿意在程序中运用一种,比如我只使用promise,不使用async和await;也有只用async和await,而不是用promise。所以现在来为大家讲讲一起共存需要注意写什么?

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

    示例

    下面这段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')
    
    JavaScript

    上述,在Chrome 66node v10中,正确输出是:

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

    知识点

    显然,这考察的是js中的事件循环和回调队列。注意以下几点:
    Promise优先于setTimeout宏任务。所以,setTimeout回调会在最后执行。
    Promise一旦被定义,就会立即执行。
    Promiserejectresolve是异步执行的回调。所以,resolve()会被放到回调队列中,在主函数执行完和setTimeout前调用。
    await执行完后,会让出线程。async标记的函数会返回一个Promise对象

    难点

    最令人困惑的,就是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 ')
      })
    }
    
    JavaScript

    流程

    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
  • 相关阅读:
    Troubleshooting FIM: (No Display Name) in FIM Portal
    FIM2010同步用户
    Problem with WinRM on Exchange 2013 Management Shell and Exchange Toolbox on a new exchange 2013 with CAFE and BE on single server installation
    SAML : A SAML stack
    Flutter-使用Custompaint绘制一个圆圈进度条
    Flutter-自绘组件CustomPaint
    AnimatedSwitcher
    入职一个月
    flutter--static关键字
    Flutter-变量初始化问题
  • 原文地址:https://www.cnblogs.com/shaozhu520/p/11341030.html
Copyright © 2011-2022 走看看