一次性让你懂async/await

一次性让你懂async/await,解决回调地狱

什么是async?

欢迎留言讨论

async 函数是 Generator 函数的语法糖。使用 关键字 async 来表示,在函数内部使用 await 来表示异步。相较于 Generatorasync 函数的改进在于下面四点:

  • 内置执行器Generator 函数的执行必须依靠执行器,而 async 函数自带执行器,调用方式跟普通函数的调用一样
  • 更好的语义asyncawait 相较于 *yield 更加语义化
  • 更广的适用性co 模块约定,yield 命令后面只能是 Thunk 函数或 Promise对象。而 async 函数的 await 命令后面则可以是 Promise 或者 原始类型的值(Number,string,boolean,但这时等同于同步操作)
  • 返回值是 Promiseasync 函数返回值是 Promise 对象,比 Generator 函数返回的 Iterator 对象方便,可以直接使用 then() 方法进行调用

此处总结参考自:理解async/await

async是ES7新出的特性,表明当前函数是异步函数,不会阻塞线程导致后续代码停止运行。

怎么用

申明之后就可以进行调用了

async function asyncFn() {
  return 'hello world';
}
asyncFn();

这样就表示这是异步函数,返回的结果

img

async 表示函数里有异步操作

await 表示紧跟在后面的表达式需要等待结果。

返回的是一个promise对象,状态为resolved,参数是return的值。那再看下面这个函数

async function asyncFn() {
    return '我后执行'
}
asyncFn().then(result => {
    console.log(result);
})
console.log('我先执行');

img

上面的执行结果是先打印出'我先执行',虽然是上面asyncFn()先执行,但是已经被定义异步函数了,不会影响后续函数的执行。

现在理解了async基本的使用,那还有什么特性呢?

async定义的函数内部会默认返回一个promise对象,如果函数内部抛出异常或者是返回reject,都会使函数的promise状态为失败reject

async function e() {    
    throw new Error('has Error');
}
e().then(success => console.log('成功', success))   
   .catch(error => console.log('失败', error));

img

我们看到函数内部抛出了一个异常,返回rejectasync函数接收到之后,判定执行失败进入catch,该返回的错误打印了出来。

async function throwStatus() {    
    return '可以返回所有类型的值'
}
throwStatus().then(success => console.log('成功', success))             
             .catch(error => console.log('失败', error));

//打印结果

成功 可以返回所有类型的值
async`函数接收到返回的值,发现不是`异常`或者`reject`,则判定成功,这里可以`return`各种数据类型的值,`false`,`NaN`,`undefined`...总之,都是`resolve

但是返回如下结果会使async函数判定失败reject

  1. 内部含有直接使用并且未声明的变量或者函数。
  2. 内部抛出一个错误throw new Error或者返回reject状态return Promise.reject('执行失败')
  3. 函数方法执行出错(🌰:Object使用push())等等…

awaited rejected后下行代码不执行

await抛出异常或者返回状态变为“rejected”后,await下面那一行代码不会继续执行了

可以使用trt catch来捕获异常,try代码块下面的代码仍是可以执行的。

  const confirm = async (e) => {
    const eTarget = e.currentTarget;
    try {
      await basicFormRef.value.submitForm();
      showDisabled(eTarget);
      if (title.value === '新增') {
        await createCallback(formData);
      } else if (title.value === '修改') {
        await updateCallback(formData);
      }
      ElMessage.success(`${message[title.value]}`);
      close();
      emit('searchData');
    } catch (error) {
      console.error(error);
    } finally {
      hiddenDisabled(eTarget);
    }
    console.log('我仍然可以执行');
  };

还有一点,在async里,必须要将结果return回来,不然的话不管是执行reject还是resolved的值都为undefine,建议使用箭头函数。

其余返回结果都是判定resolved成功执行。

//正确reject方法。必须将reject状态return出去。
async function PromiseError() {    
   return Promise.reject('has Promise Error');
}

//这是错误的做法,并且判定resolve,返回值为undefined,并且Uncaught报错
async function PromiseError() {
  Promise.reject('这是错误的做法');
}

PromiseError().then(success => console.log('成功', success))              
              .catch(error => console.log('失败', error));

img

我们看到第二行多了个Promise对象打印,不用在意,这个是在Chrome控制台的默认行为,我们平常在控制台进行赋值也是同样的效果。如果最后执行语句或者表达式没有return返回值,默认undefined,做个小实验。

var a = 1;
//undefined
------------------------------------------------------------
console.log(a);
//1
//undefined
------------------------------------------------------------
function a(){ console.log(1) }
a();
//1
//undefined
------------------------------------------------------------
function b(){ return console.log(1) }
b();
//1
//undefined
------------------------------------------------------------
function c(){ return 1}
c();
//1
------------------------------------------------------------
async function d(){
    '这个值接收不到'
}
d().then(success => console.log('成功',success));
//成功  undefined
//Promise { <resolved>: undefined }
-----------------------------------------------------------
async function e(){
    return '接收到了'
}
e().then(success => console.log('成功',success));
//成功  接收到了
//Promise { <resolved>: undefined }

最后一行Promise { <resolved> : undefined } 是因为返回的是console.log执行语句,没有返回值。

d().then(success => console.log('成功',success)}
等同于
d().then(function(success){ 
            return console.log('成功',success);
        });

js本身是单线程的,通过v8我们可以拥有"异步"的能力

认识完了async,来讲讲await。

await是什么?

await意思是async wait(异步等待)。这个关键字只能在使用async定义的函数里面使用。任何async函数都会默认返回promise,并且这个promise解析的值都将会是这个函数的返回值,而async函数必须等到内部所有的 await 命令的 Promise 对象执行完,才会发生状态改变。

打个比方,await是学生,async是校车,必须等人齐了再开车。

就是说,必须等所有await 函数执行完毕后,才会告诉promise我成功了还是失败了,执行then或者catch

async function awaitReturn() {     
    return await 1
};
awaitReturn().then(success => console.log('成功', success))
             .catch(error => console.log('失败',error))

img

在这个函数里,有一个await函数,async会等到await 1 这一步执行完了才会返回promise状态,毫无疑问,判定resolved

*很多人以为await会一直等待之后的表达式执行完之后才会继续执行后面的代码,实际上await是一个让出线程的标志await后面的函数会先执行一遍(比如await Fn()的Fn ,并非是下一行代码),然后就会跳出整个async函数来执行后面js栈的代码。等本轮事件循环执行完了之后又会跳回到async函数中等待await***后面表达式的返回值,如果返回值为非promise则继续执行async函数后面的代码,否则将返回的promise放入Promise队列(Promise的Job Queue)

来看个简单点的例子

const timeoutFn = function(timeout){ 
	return new Promise(function(resolve){
		return setTimeout(resolve, timeout);
               });
}

async function fn(){
    await timeoutFn(1000);
    await timeoutFn(2000);
    return '完成';
}

fn().then(success => console.log(success));

这里本可以用箭头函数写方便点,但是为了便于阅读本质,还是换成了ES5写法,上面执行函数内所有的await函数才会返回状态,结果是执行完毕3秒后才会弹出’完成’。

正常情况下,await 命令后面跟着的是 Promise ,如果不是的话,也会被转换成一个 立即 resolve 的 Promise。

也可以这么写

function timeout(time){
    return new Promise(function(resolve){
        return setTimeout(function(){ 
                    return resolve(time + 200)
               },time);
    })
}

function first(time){
    console.log('第一次延迟了' + time );
    return timeout(time);
}
function second(time){
    console.log('第二次延迟了' + time );
    return timeout(time);
}
function third(time){
    console.log('第三次延迟了' + time );
    return timeout(time);
}

function start(){
    console.log('START');
    const time1 = 500;
    first(time1).then(time2 => second(time2) )
                .then(time3 => third(time3)  )
                .then(res => {
                              console.log('最后一次延迟' + res );
                              console.timeEnd('END');
                             })
};
start();

这样用then链式回调的方式执行resolve

//打印结果

START
第一次延迟了500
第二次延迟了700
第三次延迟了900
最后一次延迟1100
END

用async/await呢?

async function start() {
    console.log('START');
    const time1 = 500;
    const time2 = await first(time1);
    const time3 = await second(time2);
    const res = await third(time3);
    console.log(`最后一次延迟${res}`);
    console.log('END');
}
start();

达到了相同的效果。但是这样遇到一个问题,如果await执行遇到报错呢

async function start() {
    console.log('START');
    const time1 = 500;
    const time2 = await first(time1);
    const time3 = await Promise.reject(time2);
    const res = await third(time3);
    console.log(`最后一次延迟${res}`);
    console.log('END');
}
start();

img

返回reject后,后面的代码都没有执行了,以此迁出一个例子:

let last;
async function throwError() {  
    await Promise.reject('error');    
    last = await '没有执行'; 
}
throwError().then(success => console.log('成功', last))
            .catch(error => console.log('失败',last))

img

其实async函数不难,难在错处理上。

上面函数,执行的到await排除一个错误后,就停止往下执行,导致last没有赋值报错。

async里如果有多个await函数的时候,如果其中任一一个抛出异常或者报错了,都会导致函数停止执行,直接reject;

怎么处理呢,可以用try/catch,遇到函数的时候,可以将错误抛出,并且继续往下执行。

let last;
async function throwError() {  
    try{  
       await Promise.reject('error');    
       last = await '没有执行'; 
    }catch(error){
        console.log('has Error stop');
    }
}
throwError().then(success => console.log('成功', last))
            .catch(error => console.log('失败',last))

img

这样的话,就可以继续往下执行了。

来个🌰练习下

function testSometing() {
    console.log("testSomething");
    return "return testSomething";
}

async function testAsync() {
    console.log("testAsync");
    return Promise.resolve("hello async");
}

async function test() {
    console.log("test start...");

    const testFn1 = await testSometing();
    console.log(testFn1);

    const testFn2 = await testAsync();
    console.log(testFn2);

    console.log('test end...');
}

test();

var promiseFn = new Promise((resolve)=> { 
                    console.log("promise START...");
                    resolve("promise RESOLVE");
                });
promiseFn.then((val)=> console.log(val));

console.log("===END===")

执行结果

img

我们一步步来解析

首先test()打印出test start...

然后 testFn1 = await testSomething(); 的时候,会先执行testSometing()这个函数打印出“testSometing”的字符串。

testAsync()执行完毕返回resolve,之后await会让出线程就会去执行后面的,触发promiseFn打印出“promise START...”。

接下来会把返回的Promiseresolve("promise RESOLVE")放入Promise队列(Promise的Job Queue),继续执行打印“===END===”。

等本轮事件循环执行结束后,又会跳回到async函数中(test()函数),等待之前await 后面表达式的返回值,因为testSometing() 不是async函数,所以返回的是一个字符串“return``testSometing”。

test()函数继续执行,执行到testFn2(),再次跳出test()函数,打印出“testAsync”,此时事件循环就到了Promise的队列,执行promiseFn.then((val)=> console.log(val));打印出“promise RESOLVE”。

之后和前面一样 又跳回到test函数继续执行console.log(testFn2)的返回值,打印出“hello async”。

最后打印“test end...”。

加点料,让testSomething()变成async

async function testSometing() {
    console.log("testSomething");
    return "return testSomething";
}

async function testAsync() {
    console.log("testAsync");
    return Promise.resolve("hello async");
}

async function test() {
    console.log("test start...");

    const testFn1 = await testSometing();
    console.log(testFn1);

    const testFn2 = await testAsync();
    console.log(testFn2);

    console.log('test end...');
}

test();

var promiseFn = new Promise((resolve)=> { 
                    console.log("promise START...");
                    resolve("promise RESOLVE");
                });
promiseFn.then((val)=> console.log(val));

console.log("===END===")

执行结果

img

和上一个例子比较发现promiseFn.then((val)=> console.log(val)); 先于console.log(testFn1) 执行。

原因是因为现在的版本(今天是2019-11-06 12:03:46)async函数会被await resolve,

function testSometing() {
    console.log("testSomething");
    return "return testSomething";
}
console.log(Object.prototype.toString.call(testSometing)) // [object Function]
console.log(Object.prototype.toString.call(testSometing())) // [object String]

async function testSometing() {
    console.log("testSomething");
    return "return testSomething";
}
console.log(Object.prototype.toString.call(testSometing)) // [object AsyncFunction]
console.log(Object.prototype.toString.call(testSometing())) // [object Promise]

testSomething()已经是async函数,返回的是一个Promise对象需要等它 resolve 后将当前Promise 推入队列,随后先清空调用栈,所以会"跳出" test() 函数执行后续代码,随后才开始执行该 Promise。

更新

今天是2019-11-06 12:03:46,收到几位掘友的反馈,最后一个例子的执行时机不符合现有版本的结果,原因可能是在文章完结后至今的一年半内 Chrome V8 更新导致,如果有 Opera 、其他使用了 Chrome 老版本内核的浏览器,或者到 这个地址 下载旧版本 Chrome,笔者测试使用的是 Opera,在 Chrome V8 release 文档或里并未找到相关说明。

现在版本结果是:

和之前没加async的结果一样
在这里插入图片描述

img

“Node.js 是一个开源的、跨平台的、后端的 JavaScript 运行时环境,构建在 V8 引擎之上,在浏览器之外执行 Javascript 代码。

  • 之前的版本中Node.js 8所对应的v8引擎,await 的函数需要3次tick才会被执行,v8团队为了提升性能,违反了规范,没有严格按照 promise-resolve-functions 的第13步执行,导致3次tick减少2次。原文
  • Node.js 10 实现了正确的行为,即首先执行链式处理程序,然后继续使用异步函数。(Node.js 10 中不再有 await 的 bug)
  • nodejs10之后 再次优化v8引擎 。优化避免了再次创建 Promise 包装器,在这种情况下,我们从最少三个 microtick 到只有一个 microtick。这种行为类似于 Node.js 8 所做的,但是现在它不再是一个 bug - 它现在是一个正在标准化的优化!
  • 在node11机制已经和浏览器统一,从表面上, 你大可以理解 await AsyncFunction 返回的 Promise 为同步任务。

在这里插入图片描述

node11发布时间 2018年10月24日

在这里插入图片描述

Opera 56.0.30发布时间 2018年8月2日,小于node11的发布时间,所以当时v8引擎和node10的表现是一致的

在这里插入图片描述

参考文献:理解 JavaScript 的 async/await

转自掘金大佬

  • 4
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: async/awaitJavaScript 中的异步编程机制,旨在简化异步代码的编写。使用 async/await 可以让异步代码的执行流程更加类似于同步代码,从而提高代码的可读性和可维护性。 当一个函数前面加上关键字 `async`,该函数就成为了一个异步函数。异步函数中可以使用 `await` 关键字等待一个异步操作(例如一个 Promise 的 resolve)完成,在该异步操作完成之前,函数的执行将会暂停。 例如: ``` async function fetchData() { const response = await fetch('https://api.example.com/data'); const data = await response.json(); return data; } ``` 在上面的代码中,我们使用了 `await` 关键字等待 `fetch` 函数的返回值,这意味着当我们调用 `fetchData` 函数时,代码将会暂停,直到请求完成为止。 ### 回答2: async/await 是 ECMAScript 2017 引入的一种语法糖,用于简化异步编程的写法。通过使用 async 和 await 关键字,可以将异步代码以同步的方式编写,增强代码的可读性和维护性。 async 关键字用于声明一个函数是异步的,当函数被调用时,它会返回一个 Promise 对象。在函数体内,可以使用 await 关键字来暂停函数的执行,等待 Promise 对象的状态变为 resolved,然后获取到 Promise 的结果。 使用 async/await 可以避免传统的回调地狱,代码结构更清晰、更易理解。通过 await 关键字,可以实现按照自然的顺序编写异步代码,而不必使用嵌套的回调函数。 另外,async/await 还具有错误处理的能力。可以使用 try-catch 块来捕获和处理异步操作中的异常。在 async 函数内部,如果出现异常,可以使用 throw 关键字抛出异常,然后在调用该函数的地方使用 try-catch 来捕获并处理异常。 需要注意的是,await 关键字只能在 async 函数中使用。如果在非 async 函数中使用 await,会导致语法错误。 总之,async/await 是一种语法糖,用于简化异步编程的写法。它通过 async 和 await 关键字,让异步代码以同步的方式编写,提高代码的可读性和可维护性。同时,它还具备错误处理的能力,让程序员能够更方便地处理异步操作中的异常。 ### 回答3: async awaitJavaScript中一种用来处理异步操作的语法。在过去,处理异步操作通常使用回调函数或者Promise,但是这两种方式都会导致代码的嵌套层级过深,降低代码的可读性和维护性。而async await则可以使异步代码更加简洁和易读。 使用async await,我们可以在异步函数前面加上async关键字,使其成为一个异步函数。在异步函数中,我们可以使用await关键字等待一个异步操作的结果,而不需要使用回调函数或者处理Promise的then和catch。 具体来说,async await的工作原理是将异步操作转换为一个Promise对象,然后使用await关键字等待该Promise的解决结果。当await关键字后面的Promise解决之后,将其结果返回给异步函数,并继续执行后面的代码。 使用async await可以使异步代码具有同步代码的写法风格,提高代码的可读性和可维护性。它能够避免回调地狱的问题,使异步操作更加直观和易于理解。此外,由于async await基于Promise实现,所以它也具有Promise的一些优点,比如可以很方便地进行错误处理和链式调用。 总结来说,async awaitJavaScript中用于处理异步操作的一种语法糖,通过使用async关键字定义一个异步函数,再用await关键字等待异步操作的结果,使异步代码具有同步代码的风格,提高代码的可读性和可维护性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值