Async/Await:优雅、简洁地通过同步的方式实现异步

在 JavaScript 的异步编程中,我们经常使用回调函数、Promise 和 Async/Await 来解决异步操作的问题。而 Async/Await 又是 Promise 的语法糖,它的出现让异步编程变得更加直观和易于理解。本文将详细讲解 Async/Await 如何通过同步的方式实现异步。

一、异步编程的问题

在 Web 开发中,我们经常需要进行异步操作,比如从服务器获取数据,或者执行耗时操作。这些任务通常需要一定的时间来完成,而在这段时间内,JavaScript 不能停止执行其他代码,否则会导致界面卡住或者无响应。因此,我们需要使用异步编程技术来处理这些操作。

传统的异步编程技术有回调函数和 Promise。使用回调函数时,我们需要将后续的操作写在回调函数中,这样才能确保在异步操作完成后执行。回调函数虽然简单易用,但是嵌套多个回调函数会导致代码难以阅读和维护。而 Promise 解决了这个问题,它可以链式调用,避免了回调函数嵌套的问题。但是,Promise 的语法并不是那么直观,需要一定的学习成本。

为了更加直观地表达异步代码,ES2017 引入了 Async/Await。Async/Await 可以使异步代码看起来像同步代码,这样可以使代码更加易于理解和维护。接下来,我们将详细讲解 Async/Await 的实现原理。

二、 Async/Await 的实现原理

Async 和 Await 都是异步编程的关键字。在 ES2017 中,Async 函数用来声明一个异步函数,它的定义方式类似于普通的函数,但是在函数关键字前面添加 async 关键字,如下所示:

async function fetchData() {
  // 异步操作
}

我们可以在 Async 函数内部使用 await 关键字来等待异步操作完成。await 表示等待异步操作返回结果后再继续执行后续的代码。

async function fetchData() {
  const result = await fetch("/api/data");
  console.log(result);
}

这段代码中,我们使用了 fetch 函数来获取服务器数据,fetch 返回的是 Promise 实例。我们在该 Promise 实例前面加上 await 关键字,表示等待该 Promise 对象返回数据后再继续执行后续的代码。

当 Async 函数被调用时,它返回的是一个 Promise 对象。Promise 对象有三种状态:已完成、已拒绝和未完成。如果 Async 函数内部没有抛出异常,则该 Promise 对象将进入已完成状态,并返回 Async 函数返回值;如果 Async 函数内部抛出异常,则该 Promise 对象将进入已拒绝状态,并返回抛出的异常。例如,下面这个例子中,Async 函数返回的 Promise 对象的状态为已完成,并返回字符串 “Hello World!”:

async function hello() {
  return "Hello World!";
}
hello().then((result) => console.log(result)); // 输出 "Hello World!"

在下面的示例中,我们使用 Async/Await 实现一个简单的异步操作:

async function fetchData() {
  try {
    const response = await fetch("https://jsonplaceholder.typicode.com/posts");
    const data = await response.json();
    console.log(data);
  } catch (error) {
    console.error(error);
  }
}
fetchData();

在这个例子中,我们使用了 fetch 函数来获取服务器数据,并且使用 await 等待数据返回。如果出现异常,则使用 try…catch 处理异常。这段代码使用起来非常直观和易于理解。

Async/Await 的同步实现原理

虽然使用 Async/Await 可以使异步代码看起来像同步代码,但是底层仍然是异步执行的。那么,Async/Await 是如何通过同步的方式实现异步的呢?答案就是 Generator 函数和 Promise。

Generator 函数是一种特殊的函数,它可以被暂停和恢复执行。在 Generator 函数中,我们可以使用 yield 关键字将控制权交给调用方,并在下次调用时从上次暂停的位置继续执行。这种特性可以用来实现异步操作。

Promise 是 ES6 引入的另一种异步编程技术。Promise 对象表示一个尚未完成或失败的操作,它可以被异步执行,并返回一个代表操作结果的值。

Async 函数实际上是一种特殊的 Generator 函数,它使用 yield 关键字暂停执行,并在异步操作完成后,通过调用 next 方法恢复执行。这个过程中,Async 函数内部创建了一个 Promise 对象,并将该 Promise 对象返回给调用方。下面是 Async 函数的简化版实现:

function asyncToGenerator(generatorFunc) {
  return function () {
    const generator = generatorFunc.apply(this, arguments);
    return new Promise((resolve, reject) => {
      function step(key, arg) {
        let generatorResult;
        try {
          generatorResult = generator[key](arg);
        } catch (error) {
          reject(error);
        }
        const { value, done } = generatorResult;
        if (done) {
          resolve(value);
        } else {
          Promise.resolve(value).then(
            (result) => step("next", result),
            (error) => step("throw", error)
          );
        }
      }
      step("next");
    });
  };
}

这段代码中,我们定义了一个名为 asyncToGenerator 的函数,它接收一个 Generator 函数作为参数,并返回一个 Promise 对象。在 asyncToGenerator 函数内部,我们首先调用传入的 Generator 函数,获取到一个迭代器对象。然后,我们在 Promise 对象的构造函数中使用递归调用的方式来处理每一次迭代。如果当前迭代已经完成,则调用 resolve 函数,将结果返回给调用方;否则,将该迭代的 Promise 对象通过 then 方法注册成功和失败回调函数,并在回调函数中继续处理下一次迭代。

三、Async/Await 的使用场景

Async/Await 通常用于处理多个异步操作的情况,它可以避免回调地狱和 Promise 层层嵌套的问题。下面是一个具有挑战性的使用场景。

假设我们需要获取某些商品的信息并计算它们的总价格,其中每个商品需要从服务器获取,并且需要等待前一个商品请求完成后才能发送下一次请求。在写传统异步代码时,我们可能会陷入回调地狱:

function getTotalPrice(items) {
  let totalPrice = 0;
  fetchItem(0, items);

  function fetchItem(index, items) {
    if (index >= items.length) {
      console.log("totalPrice:", totalPrice);
      return;
    }
    const item = items[index];
    fetch(`/api/items/${item.id}`).then((response) => {
      response.json().then((data) => {
        item.price = data.price;
        totalPrice += item.price * item.count;
        fetchItem(index + 1, items);
      });
    });
  }
}

这段代码中,我们首先定义了一个 getTotalPrice 函数,它接收一个商品列表作为参数,并计算所有商品的总价格。我们在该函数中定义了一个名为 fetchItem 的递归函数,用于依次获取每个商品的价格,并累加到 totalPrice 变量中。在 fetchItem 函数中,我们使用 fetch 函数获取商品信息,然后使用嵌套的 Promise.then 调用来等待异步操作返回。这段代码虽然可行,但是非常难以理解和维护。

使用 Async/Await 可以让代码更加直观和易于理解:

async function getTotalPrice(items) {
  let totalPrice = 0;
  for (let item of items) {
    const response = await fetch(`/api/items/${item.id}`);
    const data = await response.json();
    item.price = data.price;
    totalPrice += item.price * item.count;
  }
  console.log("totalPrice:", totalPrice);
}

可以看到,在上面的代码中,我们使用了 Async/Await 和 for…of 循环,避免了回调地狱和 Promise 层层嵌套的问题。这样的代码看起来非常简单和直观。

四、小结一下

Async/Await 是一种比较新的异步编程技术,它使异步代码看起来像同步代码,更加直观和易于理解。Async/Await 的实现原理是 Generator 函数和 Promise,它通过同步的方式实现异步。使用 Async/Await 可以避免回调地狱和 Promise 层层嵌套的问题。Async/Await 通常用于处理多个异步操作的情况,这样的代码看起来非常简单和直观。

原文链接:https://juejin.cn/post/7245922875182170172 作者:Cosolar

(0)
上一篇 2023年6月19日 上午11:08
下一篇 2023年6月20日 上午10:00

相关推荐

发表回复

登录后才能评论