写点什么

2K 字就能理解的 async/await 原理,还要拖多久?

作者:梁木由
  • 2023-02-01
    北京
  • 本文字数:3750 字

    阅读完需:约 12 分钟

2K字就能理解的async/await原理,还要拖多久?

前言

上篇文章5K字 由浅入深聊聊Promise实现原理,中讲述了 Promise 内部的实现原理。今天来聊聊asyncawait,那么 async 与 await 到底是什么呢。都说是语法糖,就来深入理解下 async/await 吧


来看下 MDN 的概念

async 函数是使用async关键字声明的函数。async 函数是 AsyncFunction 构造函数的实例,并且其中允许使用 await 关键字

await 操作符用于等待一个 Promise 兑现并获取它兑现之后的值。它只能在异步函数或者模块顶层中使用。

asyncawait 关键字让我们可以用一种更简洁的方式写出基于 Promise 的异步行为,而无需刻意地链式调用 promise

async

async在字面上的意思呢,是异步的概念,根据 MDN 的概念呢,说明 async 声明的是一个异步构造函数,来看如下示例


const fn1 = async function fn(){  return 1}console.log(fn1())// Promise {<fulfilled>: 1}
复制代码


根据上述示例内容,表述 async 声明了一个异步构造函数,并且调用了该函数,返回结果是一个Promise对象。


那问题来了,在看上述代码异步函数中 return 的是 1,结果却是一个Promise对象,不急,答案来了。


如果在函数中 return 的不是一个 promise,那么将等同于使用 Promise.resolve(x)给包装起来


function fn() {  return Promise.resolve(1);}
复制代码


将常规函数转换成 Promise,返回值也是一个 Promise 对象


那这么看 async 与 Promise 有什么区别呢,看着是没什么区别,先不着急,再接着看下await

await

await字面意思呢等待等候的意思,那到底是在等什么呢,等 promise 对象吗?


const fn1 = function fn() {  return Promise.resolve(1);};async function result() {  const r1 = await fn1();  console.log(r1); // 1}result();
复制代码


还可以等其它值吗?


const fn1 = function fn() {  return Promise.resolve(1);};const fn2 = function test() {  return "test";};async function result() {  const r1 = await fn1();  const r2 = await fn2();  console.log(r1, r2); // 1 'test'}result();
复制代码


结果呢,不是 promise 对象的值也等到了。


await 不仅仅用于等 Promise 对象,它可以等任意表达式的结果。并且返回处理结果

只能在 async 函数内部使用

async/await 作用

根据上述内容呢,了解到 async 与 await 如何使用以及返回结果,那它们的作用体现到哪了呢?


先看下列有一个业务,业务内容呢需要重复请求操作,但是接口参数呢都需要在上一个请求结果中获取,先看下例子


function getFetch(type) {  setTimeout(() => {    let result = type + 1;    return result;  }, 1000);}getFetch(a).then((b) => {  getFetch(b).then((c) => {    getFetch(c).then((data) => {      return data;    });  });});
复制代码


多重的异步操这是不是传说中的回调地狱呢,那怎么解决呢


用 promise 方法来解决


function getFetch(type) {  return new Promise((resolve, reject) => {    let result = type + 1;    resolve(result);  });}
getFetch(0) .then((res) => { console.log(res); return getFetch(res); }) .then((res) => { console.log(res); return getFetch(res); }) .then((res) => { console.log(res); });
复制代码


来用 async/await 来解决


function getFetch(type) {  let result = type + 1;  return result;}
async function getResult(a) { const n1 = await getFetch(a); const n2 = await getFetch(n1); const n3 = await getFetch(n2); console.log(n1, n2, n3);}getResult(0);
复制代码


输出结果呢与Promise解决方案是一致的,但是代码看起来简洁明了


用同步方式,执行异步操作达到排队效果,解决回调地狱

Generator

async/await 为什么说是语法糖呢,是谁的语法糖呢?


在阮一峰的 ES6 入门教程中有说到:


async 函数是什么?一句话,它就是 Generator 函数的语法糖。


Generator 函数就是一个封装的异步任务,或者说是异步任务的容器。

异步操作需要暂停的地方,都用 yield 语句注明

调用 Generator 函数,返回的是指针对象(这是它和普通函数的不同之处),。调用指针对象的 next 方法,会移动内部指针。

next 方法的作用是分阶段执行 Generator 函数。每次调用 next 方法,会返回一个对象,表示当前阶段的信息( value 属性和 done 属性)。value 属性是 yield 语句后面表达式的值,表示当前阶段的值;done 属性是一个布尔值,表示 Generator 函数是否执行完毕,即是否还有下一个阶段。


了解 generator 用法


function* Generator() {  yield "1";  yield Promise.resolve(2);  return "3";}var gen = Generator();console.log(gen); //返回一个指针对象 Generator {<suspended>}
复制代码


调用 next 方法


let res1 = gen.next();console.log(res1); // 返回当前阶段的值 { value: '1', done: false }
let res2 = gen.next();console.log(res2); // 返回当前阶段的值 { value: Promise { 2 }, done: false }
res2.value.then((res) => { console.log(res); // 2});
let res3 = gen.next();console.log(res3); // { value: '3', done: true }
let res4 = gen.next();console.log(res4); // { value: undefined, done: true }
复制代码

实现 async/await

async/await 的理解


  • async 函数执行结果返回的是一个 Promise

  • async 函数就是将 Generator 函数的星号(*)替换成 async,将 yield 替换成 await

  • async/await 就是 Generator 的语法糖,其核心逻辑是迭代执行 next 函数


先来初步实现一个执行结果返回 Promise 的函数


function muYouAsync(){    // 返回一个函数    return function(){        // 返回一个promise        return new Promise((resolve, reject) => {
}) }}
复制代码


并且呢 muYouAsync 接受一个 Generator 函数作为参数的,那我们再来完善一下


function* gen() {
}//接受一个Generator函数作为参数function muYouAsync(gen){ // 返回一个函数 return function(){ // 返回一个promise return new Promise((resolve, reject) => {
}) }}
复制代码


我们来测试下看下执行结果是否返回 Promise


const asyncGen = muYouAsync(gen)console.log(asyncGen()) //Promise {<pending>}
复制代码


看输出结果的话与执行结果返回 Promise 是一致的


至此呢第一部分函数执行返回结果已完成,那我们来完善一下 Generator 函数


const getFetch = (nums) =>  new Promise((resolve) => {    setTimeout(() => {      resolve(nums + 1);    }, 1000);  });
function* gen() { let res1 = yield getFetch(1); let res2 = yield getFetch(res1); let res3 = yield getFetch(res2); return res3;}
复制代码


Generator 函数也编写完成了,在下一步我们要想,怎么让它执行起来了呢,那就需要用到 Generator 核心逻辑迭代执行 next 函数,注意点是需要将 next 迭代执行


//接受一个Generator函数作为参数function muYouAsync(gen) {  // 返回一个函数  return function () {    // 返回一个promise    return new Promise((resolve, reject) => {      // 执行Generator函数      let g = gen();      const next = (context) => {        const { done, value } = g.next(context);        if (done) {          // 这时候说明已经是完成了,需要返回结果          resolve(value);        } else {          // 继续执行next函数,传入执行结果          return Promise.resolve(value).then(val => next(val))        }      };      next();    });  };}
复制代码


整体的逻辑已经全都补充好了,那我们还需要在完善下最后一步,async 返回的是 promise,所以我们可以用 try catch 来捕获

完整代码

//接受一个Generator函数作为参数function muYouAsync(gen) {  // 返回一个函数  return function () {    // 返回一个promise    return new Promise((resolve, reject) => {      // 执行Generator函数      let g = gen();      const next = (context) => {        let res        try {            res = g.next(context);        } catch (error) {            reject(error)        }        if (res.done) {          // 这时候说明已经是完成了,需要返回结果          resolve(res.value);        } else {          // 继续执行next函数,传入执行结果          return Promise.resolve(res.value).then(val => next(val), err => next(err))        }      };      next();    });  };}
复制代码


那我们最后来测试一下


const getFetch = (nums) =>  new Promise((resolve) => {    setTimeout(() => {      resolve(nums + 1);    }, 1000);  });
function* gen() { let res1 = yield getFetch(1); let res2 = yield getFetch(res1); let res3 = yield getFetch(res2); return res3;}
const asyncGen = muYouAsync(gen);asyncGen().then(res => {console.log(res)}); // 4
复制代码

结语

今年想法呢,是输出一些文章,总不能干了几年活了,还是碌碌无为,躺平过日子。年前那几天也刷了不少文章,看到了很多平台的前端打工人,都很卷的样子,那我今年也就卷一下子吧,腰断了就不卷了。


如果感觉此文稍微有些帮助的话,请不吝点个赞🥺🥺🥺


欢迎大家到大家庭摸鱼交流学习,争当卷王沸点


发布于: 刚刚阅读数: 3
用户头像

梁木由

关注

公众号:前端新气象 2023-01-16 加入

一个有想头的前端,对2023年充满希望、怀抱期待,相信自己做个自律的人。要学习,拒绝躺平,从我做起。⛽️

评论

发布
暂无评论
2K字就能理解的async/await原理,还要拖多久?_前端_梁木由_InfoQ写作社区