如何在forEach内使用异步调用 async/await

如何,foreach,使用,异步,调用,async,await · 浏览次数 : 1555

小编点评

**1、`forEach` 循环无法使用 `async` 和 `await`:** `forEach` 循环不包含异步代码,因此无法使用 `async` 和 `await`关键字。 **2、`async` 关键字用于定义异步函数:** `async` 关键字用于定义一个异步函数,它返回一个 `Promise`。 **3、`await` 语句用于等待异步操作完成:** `await` 语句用于等待异步操作完成,并将结果存储在一个变量中。 **4、`async` 关键字在 `forEach` 中的使用:** 在 `forEach` 循环中使用 `async` 关键字可以将异步函数包装成一个异步函数,并使用 `await` 语句等待其完成。 **5、`await` 语句用于等待 Promise 处理完毕:** `await` 语句可以用于等待多个 Promise 处理完毕,并将结果存储在一个数组中。 **示例:** 使用 `async` 和 `await` 关键字的示例: ```javascript const users = await getUsers(); users.forEach(async (user) => { const details = await user.getDetails(); console.log(details); }); ``` **其他方法:** * 使用 `for` 循环:`for (let i = 0; i < users.length; i++) {...}` * 使用 `for ... of` 的简化版:`for (const user of users) {...}` * 使用 `Promise.all` 等方法等待多个 Promise 处理完毕。

正文

翻自: How to use async and await in a forEach JS loop?
https://learn.coderslang.com/0144-how-to-use-async-and-await-in-a-foreach-js-loop/

事实上我们无法在 forEach 循环内使用 async/await 起到异步作用,让我们看看如何解决修复它。

async/await 在forEach 中为啥不起作用?

当你在forEach 循环内调用异步函数,下一个循环并不会等到上个循环结果后再被调用

想象一下,你有一个 getUsers 方法返回用户列表 User , user只有用户列表,但不包含具体的detail信息,所以你使用forEach 试图去获取用户详情

const users = await getUsers();

users.forEach(user => {
  const details = user.getDetails();
  console.log(details);
})

打印结果却是 3 个pending状态的 promise

你很自然的想到getDetails是个异步函数,需要给它加上await 关键字

const users = await getUsers();

users.forEach(user => {
  const details = await user.getDetails();
  console.log(details);
})

运行后发现报了另一个语法错误

SyntaxError: await is only valid in async function

你意识到给如果 forEach 传递的函数有是异步 则需要用 async 标记为异步函数

再次修复它

const users = await getUsers();

users.forEach(async (user) => {
  const details = await user.getDetails();
  console.log(details);
})

很好,你盯着半天发现并没有按users数组的顺序输出结果

解决方案:

你需要使用for 代替 forEach 循环重写这部分代码

const users = await getUsers();

for (let i = 0; i < users.length; i++) {
  const details = await user.getDetails();
  console.log(details);
}

也可以使用 for .... of

const users = await getUsers();

for (const user of users) {
  const details = await user.getDetails();
  console.log(details);
}

如果你实在不喜欢简单的代码,那么可以用个小技巧 使用 Array 的 map 方法

const users = await getUsers();

const userDetailsPromises = users.map(user => user.getDetails());
const userDetails = await Promise.all(userDetailsPromises);

console.log(userDetails);

让我们一步步分析
1、第一行代码没有改变就是获取 users 数组
2、使用map循环调用 getDetails 返回新的 userDetailsPromises 数组,数组元素都为Promise类型
3、使用 await Promise.all 等待所有 Promise 处理完毕
4、最后打印出结果数组

代码也可以组合简化为

const userDetails = await Promise.all(users.map(user => user.getDetails()));

与如何在forEach内使用异步调用 async/await 相似的内容:

如何在forEach内使用异步调用 async/await

翻自: How to use async and await in a forEach JS loop? https://learn.coderslang.com/0144-how-to-use-async-and-await-in-a-foreach-js-loop/ 事实上我们无法在 forEa

.NET周报【1月第2期 2023-01-13】

国内文章 【ASP.NET Core】按用户等级授权 https://www.cnblogs.com/tcjiaan/p/17024363.html 本文介绍了ASP.NET Core如何按照用户等级进行授权。 在 C# 9 中使用 foreach 扩展 https://www.cnblogs.co

Vue3 如何接入 i18n 实现国际化多语言

如何在现有 Vue 3.0 + Vite 项目中,引入 i18n 实现国际化多语言,可以手动切换,SEO友好,且完整可用的解决方案。

如何在 Windows 使用 Podman Desktop 取代 Docker Desktop

Podman Desktop 是 Docker Desktop 的免费替代品,是本地开发使用的另一个绝佳选择。它提供了类似的功能集,同时保持完全开源,让您避免使用 Docker 产品的许可问题。在本文中,您将学习如何安装和开始使用 Podman Desktop 来运行容器并部署到 Kubernete

如何在Spring Boot框架下实现高效的Excel服务端导入导出?

前言 Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置。今天我们就使用纯前对按表格控件带大家了解,如何在Spring Boot框架下实现Excel服务端导

如何更改.NET中的默认时区?

除了"在操作系统中修改时区信息,然后重启.NET应用程序,使其生效"之外。如何在不修改操作系统时区的前提下,修改.NET中的默认时区呢? 这是一位 同学兼同事 于5月21日在技术群里问的问题,我当时简单地研究了一下,就写出来了。 现在写文章分享给大家,虽然我觉得这种需求非常小众,几乎不会有人用到。

如何在Spring Boot中配置MySQL数据库连接数

1.如何在Spring Boot中配置MySQL数据库的连接数 1.1主要配置 在Spring Boot中配置MySQL数据库连接数通常涉及到两个主要的配置: (1)数据源配置:这通常是在application.properties或application.yml文件中完成的,用于设置数据源的基本参

如何在低代码平台中引用 JavaScript ?

引言 在当今快速发展的数字化时代,企业对业务应用的需求日益复杂且多元。低代码开发平台作为一个创新的解决方案,以直观易用的设计理念,打破了传统的编程壁垒,让非技术人员也能轻松构建功能完备的Web应用程序,无需深入编码。这一特性极大地简化了应用开发流程,加速了业务需求转化为实际应用的速度,为企业带来了前

如何在 K8S 集群范围使用 imagePullSecret?

在这篇文章中,我将向你展示如何在 Kubernetes 中使用 imagePullSecrets。

如何使用 Blackbox Exporter 监控 URL?

如何在 Kubernetes 中使用 Blackbox Exporter 与 Prometheus 进行 URL 监控采集、展示和告警。