Maison  >  Article  >  interface Web  >  Explication détaillée de la façon d'utiliser async/await en JavaScript

Explication détaillée de la façon d'utiliser async/await en JavaScript

青灯夜游
青灯夜游avant
2020-11-19 18:04:3816139parcourir

Explication détaillée de la façon d'utiliser async/await en JavaScript

Le async/await introduit dans ES8 est une excellente amélioration de la programmation asynchrone en JavaScript. Il fournit un moyen d'accéder à resoruces de manière asynchrone en utilisant un code de style synchrone sans bloquer le thread principal. Cependant, ils présentent également certains pièges et problèmes. Dans cet article, async/await sera exploré sous différents angles et montrera comment utiliser les frères correctement et efficacement.

Connaissances préalables

Quel est le rôle de l'async

De MDN nous pouvons voir :

async Quelle est la fonction renvoie un objet A Promise. La fonction async (y compris l'instruction de fonction, l'expression de fonction, l'expression Lambda) renverra un objet Promise Si return est une quantité directe dans la fonction, async. Cette valeur directe sera encapsulée dans un objet Promise.resolve()Promise via .

Si la fonction async n'a pas de valeur de retour, elle renverra Promise.resolve(undefined).

Quel est le rôle de wait

Appris de MDN :

await attend une expression. le résultat de l'évaluation est un objet Promise ou une autre valeur (en d'autres termes, await peut attendre le résultat de n'importe quelle expression).

Si ce qu'il attend n'est pas un objet Promise, le résultat de l'expression wait est ce qu'il attend.

S'il attend un objet Promise, wait sera occupé. Il bloquera le code suivant, attendra que l'objet Promise soit résolu, puis obtiendra la valeur résolue comme résultat de l'expression wait.

C'est pourquoi wait doit être utilisé dans les fonctions asynchrones. Les appels de fonction asynchrone ne provoqueront pas de blocage. Tous les blocages qu'ils contiennent sont encapsulés dans un objet Promise et exécutés de manière asynchrone.

Avantages de async/await

async/await L'avantage le plus important qu'il nous apporte est le style de programmation synchrone. Regardons un exemple :

// async/await
async getBooksByAuthorWithAwait(authorId) {
  const books = await bookModel.fetchAll();
  return books.filter(b => b.authorId === authorId);
}// promise
getBooksByAuthorWithPromise(authorId) {
  return bookModel.fetchAll()
    .then(books => books.filter(b => b.authorId === authorId));
}

Évidemment, la version async/await est plus facile à comprendre que la version promise. Si vous omettez le mot-clé await, le code ressemblera à n'importe quel autre langage synchrone, tel que Python.

La meilleure partie n’est pas seulement la lisibilité. async/await À ce jour, tous les principaux navigateurs prennent entièrement en charge la fonctionnalité asynchrone.

Explication détaillée de la façon dutiliser async/await en JavaScript

La prise en charge native du navigateur signifie que vous n'avez pas besoin de changer de code. Plus important encore, cela facilite le débogage. Lorsque vous définissez un point d'arrêt au point d'entrée de la fonction et que vous franchissez la ligne await, vous verrez le débogueur s'arrêter un instant pendant que bookModel.fetchAll() exécute sa tâche, puis il passera à la ligne .filter suivante, qui est plus rapide que la ligne . Le code promise est beaucoup plus simple, à l'intérieur de la promise un autre point d'arrêt doit être défini sur la ligne .filter.

Explication détaillée de la façon dutiliser async/await en JavaScript

Un autre avantage moins évident est le mot-clé async. asyncDéclarer getBooksByAuthorWithAwait() La valeur de retour de la fonction est garantie comme étant une promesse, afin que l'appelant puisse utiliser en toute sécurité getBooksByAuthorWithAwait().then(...) ou attendre getBooksByAuthorWithAwait() . Considérons l'exemple suivant (mauvaise pratique !) :

getBooksByAuthorWithPromise(authorId) {
  if (!authorId) {
    return null;
  }
  return bookModel.fetchAll()
    .then(books => books.filter(b => b.authorId === authorId));
}

Dans le code ci-dessus, getBooksByAuthorWithPromise peut renvoyer promise (normalement) ou null Value (dans des circonstances exceptionnelles ), dans des circonstances exceptionnelles, l'appelant ne peut pas appeler .then(). Avec la déclaration async, cette situation ne se présentera pas.

async/wait peut être trompeur

Certains articles comparent async/wait à Promise et prétendent qu'il s'agit de la prochaine génération de style de programmation asynchrone JavaScript , avec lequel l’auteur est profondément en désaccord. async/await est une amélioration, mais ce n'est que du sucre syntaxique et ne changera pas complètement notre style de programmation.

Essentiellement, la fonction async est toujours une promesse. Avant d'utiliser correctement la fonction async, vous devez d'abord comprendre la promesse. Ce qui est pire, la plupart du temps, vous devez utiliser en même temps que les promesses fonction async.

Considérez les fonctions

getBooksByAuthorWithAwait() et getbooksbyauthorwithpromise() de l'exemple ci-dessus. Notez que non seulement ils sont fonctionnellement identiques, mais ils ont exactement la même interface !

Cela signifie que

getbooksbyauthorwithwait() renverra une promesse, vous pouvez donc utiliser façon de l'appeler.

嗯,这未必是件坏事。只有 await 的名字给人一种感觉,“哦,太好了,可以把异步函数转换成同步函数了”,这实际上是错误的。

async/await

那么在使用 async/await 时可能会犯什么错误呢?下面是一些常见的例子。

太过串行化

尽管 await 可以使代码看起来像是同步的,但实际它们仍然是异步的,必须小心避免太过串行化。

async getBooksAndAuthor(authorId) {
  const books = await bookModel.fetchAll();
  const author = await authorModel.fetch(authorId);
  return {
    author,
    books: books.filter(book => book.authorId === authorId),
  };
}

上述代码在逻辑上看似正确的,然而,这是错误的。

  1. await bookModel.fetchAll() 会等待 fetchAll() 直到 fetchAll() 返回结果。
  2. 然后 await authorModel.fetch(authorId) 被调用。

注意,authorModel.fetch(authorId) 并不依赖于 bookModel.fetchAll() 的结果,实际上它们可以并行调用!然而,用了 await,两个调用变成串行的,总的执行时间将比并行版本要长得多得多。

下面是正确的方式:

async getBooksAndAuthor(authorId) {
  const bookPromise = bookModel.fetchAll();
  const authorPromise = authorModel.fetch(authorId);
  const book = await bookPromise;
  const author = await authorPromise;
  return {
    author,
    books: books.filter(book => book.authorId === authorId),
  };
}

更糟糕的是,如果你想要一个接一个地获取项目列表,你必须依赖使用 promises:

async getAuthors(authorIds) {
  // WRONG, this will cause sequential calls
  // const authors = _.map(
  //   authorIds,
  //   id => await authorModel.fetch(id));// CORRECT
  const promises = _.map(authorIds, id => authorModel.fetch(id));
  const authors = await Promise.all(promises);
}

简而言之,你仍然需要将流程视为异步的,然后使用 await 写出同步的代码。在复杂的流程中,直接使用 promise 可能更方便。

错误处理

promise中,异步函数有两个可能的返回值: resolvedrejected。我们可以用 .then() 处理正常情况,用 .catch() 处理异常情况。然而,使用 async/await方式的,错误处理可能比较棘手。

try…catch

最标准的(也是作者推荐的)方法是使用 try...catch 语法。在 await 调用时,在调用 await 函数时,如果出现非正常状况就会抛出异常,await 命令后面的 promise 对象,运行结果可能是 rejected,所以最好把await 命令放在 try...catch 代码块中。如下例子:

class BookModel {
  fetchAll() {
    return new Promise((resolve, reject) => {
      window.setTimeout(() => { reject({'error': 400}) }, 1000);
    });
  }
}// async/await
async getBooksByAuthorWithAwait(authorId) {
try {
  const books = await bookModel.fetchAll();
} catch (error) {
  console.log(error);    // { "error": 400 }
}

在捕捉到异常之后,有几种方法来处理它:

  • 处理异常,并返回一个正常值。(不在 catch 块中使用任何 return 语句相当于使用 return undefined,undefined 也是一个正常值。)
  • 如果你想让调用者处理它,你可以直接抛出普通的错误对象,如 throw errorr,它允许你在 promise 链中使用 async getBooksByAuthorWithAwait() 函数(也就是说,可以像getBooksByAuthorWithAwait().then(...).catch(error => ...) 处理错误); 或者可以用 Error 对象将错误封装起来,如 throw new Error(error),当这个错误在控制台中显示时,它将给出完整的堆栈跟踪信息。
  • 拒绝它,就像 return Promise.reject(error) ,这相当于 throw error,所以不建议这样做。

使用 try...catch 的好处:

  • 简单,传统。只要有Java或c++等其他语言的经验,理解这一点就不会有任何困难。
  • 如果不需要每步执行错误处理,你仍然可以在一个 try ... catch 块中包装多个 await 调用来处理一个地方的错误。

这种方法也有一个缺陷。由于 try...catch 会捕获代码块中的每个异常,所以通常不会被 promise 捕获的异常也会被捕获到。比如:

class BookModel {
  fetchAll() {
    cb();    // note `cb` is undefined and will result an exception
    return fetch('/books');
  }
}try {
  bookModel.fetchAll();
} catch(error) {
  console.log(error);  // This will print "cb is not defined"
}

运行此代码,你将得到一个错误  ReferenceError: cb is not defined。这个错误是由console.log()打印出来的的,而不是 JavaScript 本身。有时这可能是致命的:如果 BookModel 被包含在一系列函数调用中,其中一个调用者吞噬了错误,那么就很难找到这样一个未定义的错误。

让函数返回两个值

另一种错误处理方法是受到Go语言的启发。它允许异步函数返回错误和结果。详情请看这篇博客文章:

How to write async await without try-catch blocks in Javascript

简而言之,你可以像这样使用异步函数:

[err, user] = await to(UserModel.findById(1));

作者个人不喜欢这种方法,因为它将 Go 语言的风格带入到了 JavaScript 中,感觉不自然。但在某些情况下,这可能相当有用。

使用 .catch

这里介绍的最后一种方法就是继续使用 .catch()

回想一下 await 的功能:它将等待 promise 完成它的工作。值得注意的一点是 promise.catch() 也会返回一个 promise ,所以我们可以这样处理错误:

// books === undefined if error happens,
// since nothing returned in the catch statement
let books = await bookModel.fetchAll()
  .catch((error) => { console.log(error); });

这种方法有两个小问题:

  • 它是 promises 和 async 函数的混合体。你仍然需要理解 是promises  如何工作的。
  • 错误处理先于正常路径,这是不直观的。

结论

ES7引入的  async/await 关键字无疑是对J avaScrip t异步编程的改进。它可以使代码更容易阅读和调试。然而,为了正确地使用它们,必须完全理解 promise,因为 async/await 只不过是 promise 的语法糖,本质上仍然是 promise

英文原文地址:https://hackernoon.com/javascript-async-await-the-good-part-pitfalls-and-how-to-use-9b759ca21cda

更多编程相关知识,请访问:编程入门!!

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer