😲完了完了,forEach异步执行,怎么后面的先完成了!?

简介: 代码review,业务里的代码千奇百怪,到底还能遇到什么呢?oh no,真的有人在forEach里用异步调用!

嗨,大家好!这里是道长王jj~ 🎩🧙‍♂️

前端团队在对接各类接口的时候,80%的时间都在处理类似于下面的数组对象:

[{
   
    name: 'name01',
    age: '18',
    ...
}, {
   
    name: 'name02',
    age: '18',
    ...
}{
   
    name: 'name03',
    age: '18',
    ...
}, {
   
    name: 'name04',
    age: '18',
    ...
}]

而JavaScript的高级函数中就自带forEach这个遍历处理数组的方法。所以我们前端团队成员习惯性的会大量采用此方法对来源数据进行二次处理。

🏢 某天,我正在审查团队成员小明的代码时,发现了一个常见的问题:他在使用forEach方法处理异步操作时,并没有考虑到 forEach 不能保证异步任务执行顺序的问题。这给项目带来了潜在的风险和不可预测的结果。

❗review问题:forEach不能胜任异步任务!

小明在后台管理系统的开发中负责处理不同权限角色的功能。

为了快速实现不同角色的功能,他使用了forEach方法遍历角色列表并执行异步操作。然而,这种做法并不能保证异步任务按照预期的顺序执行。

下面是小明的代码片段:

async function processRoles() {
   
    let roles = ['admin', 'editor', 'employee'];
    roles.forEach(async role => {
   
            const result = await performTask(role);
            console.log(result);
    });
    console.log('任务处理完成');
}

function performTask(role) {
   
    return new Promise((resolve, reject) => {
   
        // 模拟异步操作
        setTimeout(() => {
   
                resolve(`已处理角色:${
     role}`);
        }, Math.random() * 1000);
    });
}

processRoles();

小明期望的输出结果是:

已处理角色:admin
已处理角色:editor
已处理角色:employee
任务处理完成

然而,实际的输出结果却是不确定的,可能会类似于以下情况:

任务处理完成
已处理角色:employee
已处理角色:admin
已处理角色:editor

😕 那有没有办法教教它乖乖排队呢?让我们看看该如何处理。

🧐问题溯源:forEach凭啥“随心所欲”

要理解这个问题的原因,我们需要了解forEach方法的底层实现方式。

forEach方法的核心逻辑如下:

for (var i = 0; i < length; i++) {
   
  if (i in array) {
   
    var element = array[i];
    callback(element, i, array);
  }
}

forEach方法直接遍历数组并执行回调函数,无法保证异步任务的执行顺序。如果后面的任务执行时间较短,就可能在前面的任务之前完成执行。

这就像在公司里,新同事按照名字的字母顺序逐个拜访每个同事,但每个同事的工作任务都不一样。如果某个同事恰好很快就完成了工作,而其他同事还在忙碌,那么新同事就会跟着顺序走,没有办法按照预期的顺序与同事们进行交流。

😔 这可让人头疼了!那有没有解决方案呢?

📌解决方案:for...of才能保证任务排队稳稳哒

为了解决这个问题,我们可以采用更可靠的方法,即使用for...of循环来确保异步任务按照预期顺序执行。

以下是改进后的代码示例:

async function processRoles() {
   
    let roles = ['admin', 'editor', 'employee'];
    for (const role of roles) {
   
        const result = await performTask(role);
        console.log(result);
    }
    console.log('任务处理完成');
}

通过使用for...of循环,我们可以确保异步任务按照预期顺序执行,避免了不确定性。

🚀 现在问题解决了!他重构了代码,使用了for...of循环来确保异步任务按照预期顺序执行。我们重新审查了他的代码,这次一切都按照预期运行,没有出现任何问题。💡

📖解决原理:迭代器出马,让任务听话

for...of循环实际上是基于迭代器(Iterator)的遍历方式。对于数组来说,它是一种可迭代对象,可以通过迭代器进行遍历。

我们可以通过以下方式获取数组的迭代器:

let roles = ['admin', 'editor', 'employee'];
let iterator = roles[Symbol.iterator]();
console.log(iterator.next());
console.log(iterator.next());
console.log(iterator.next());
console.log(iterator.next());

迭代器返回的结果具有valuedone属性,这使得我们可以使用迭代器来确保异步任务的顺序执行。

因此,我们的代码可以用迭代器进行如下组织:

async function processRoles() {
   
    let roles = ['admin', 'editor', 'employee'];
    let iterator = roles[Symbol.iterator]();
    let res = iterator.next();
    while(!res.done) {
   
        let value = res.value;
        console.log(value);
        console.log(await performTask(value));
        res = iterator.next();
    }
    console.log('任务处理完成');
}
processRoles()

输出结果如下:

admin
已处理角色:admin
editor
已处理角色:editor
employee
已处理角色:employee
任务处理完成

通过以上例子重新认识生成器(Generator)作为迭代器的特性,我们可以更深入地理解了for...of循环的原理和工作方式。

🎉 希望本文能够帮助你解决异步代码中按顺序执行的问题。如果你有任何疑问或者想进一步讨论相关话题,请随时告诉我。🚀✨

目录
相关文章
|
3月前
|
编译器 数据处理 C#
C#中的异步流:使用IAsyncEnumerable<T>和await foreach实现异步数据迭代
【1月更文挑战第10天】本文介绍了C#中异步流的概念,并通过使用IAsyncEnumerable<T>接口和await foreach语句,详细阐述了如何异步地迭代数据流。异步流为处理大量数据或需要流式处理数据的场景提供了一种高效且非阻塞性的方法,使得开发者能够更优雅地处理并发和数据流问题。
|
16天前
|
JavaScript 前端开发
foreach、for in和for of的区别?
foreach、for in和for of的区别?
|
2月前
|
Java 测试技术
|
4月前
|
监控 前端开发 JavaScript
async/await:使用同步的方式去写异步代码
async/await:使用同步的方式去写异步代码
53 1
|
5月前
|
JavaScript 前端开发 索引
foreach、for in 和for of的区别?
foreach、for in 和for of的区别?
|
5月前
同步和异步[多线程的异步执行操作]
同步和异步[多线程的异步执行操作]
22 0
|
10月前
|
前端开发
forEach 如果传入异步回调如何保证并行执行?
forEach 本身是同步的,但是如果回调函数是异步的,那么forEach 会立即执行下一个任务,而不会等待回调函数执行完毕,这个时候如何保证异步任务的串行执行呢?
155 0
|
Java Spring 容器
你知道 @Async 是怎么让方法异步执行的吗?
@Async 是通过注解标记来开启方法的异步执行的;对于注解的底层实现,除了 java 原生提供那种依赖编译期植入的之外,其他的基本都差不多,即运行时通过反射等方式拦截到打了注解的类或者方法,然后执行时进行横切拦截;另外这里还有一个点就是方法异步执行,所以对于 @Async 的剖析,就一定绕不开两个基本的知识点,就是代理和线程池。 在了解到这些之后,我们来拆解下 @Async 的基本原理。
166 0
你知道 @Async 是怎么让方法异步执行的吗?
|
PHP 开发者
Foreach 循环|学习笔记
快速学习 Foreach 循环
110 0
Foreach 循环|学习笔记
|
JavaScript 索引
js中forEach的用法、forEach如何跳出循环、forEach与for之间的区别
js中forEach的用法、forEach如何跳出循环、forEach与for之间的区别