Vuejs 记录踩坑:await 不能在foreach里面使用

记录踩坑

这篇文章详细讲解了为什么await 不能在foreach里面使用,以及使用方法。

参考

https://juejin.im/post/5cb5a669e51d456e845b41f6

 

简单结论:

如果想使用await ,最好搭配 for 。

而forEach是一个的回调函数,也就是说这些回调函数会立即执行。

foreach里面需要搭配其他使用方法。

 

在 JavaScript 中,`forEach` 并不是一个适合直接结合 `async/await` 使用的工具,因为它的设计并没有考虑到对异步操作的支持。当你尝试在一个 `forEach` 回调函数加上 `await` 时,可能会遇到一些奇怪的行为甚至抛出错误。这是因为虽然你在回调函数前面加了 `async` 关键字将其变成了异步函数,但是 `forEach` 自身并不会等待这些异步任务结束。 ### 错误原因 当我们在 `forEach` 内部使用 `await` 的时候,实际上只是让该迭代步骤内的代码暂停执行,而整个 `forEach` 方法本身仍然是立即返回并且不会阻塞接下来的代码。 例如下面这个例子: ```javascript const dataArray = [1, 2, 3]; dataArray.forEach(async (item) => { await someAsyncFunction(item); // 即使这用了await,外部也不会等它完成 }); console.log('This will run before all async calls are done.'); ``` 尽管我们有 `await`,但由于 `forEach` 不会真正等待那些 promise 全部 resolve,所以最终会出现未预期的结果——即“没等到”所有异步调用都完成了便继续往下执行程序流。 --- ### 正确做法 为了更有效地控制异步流程并确保按照期望顺序运行每一项任务,我们可以选择其他替代方案如简单的for-of循环或是Promise-based的方法: #### 方案一:改用普通 `for...of` 循环代替 `forEach` 这是最简单也是推荐的方式之一。与 `forEach` 相比,标准的 `for...of` 可完美配合 `async/await` 实现逐个串行执行的效果。 ```javascript const items = [/* your array */]; for(const item of items){ try{ const result = await fetchSomethingRelatedToItem(item); processResult(result); } catch(error){ handleError(error,item); } } // 下一步代码将在此处准确无误地被执行,前提是上述每轮均已完成 performNextStepAfterAllProcessed(); ``` 这样就能保证每个请求都被依次发出,并且直到上一个成功后再进入下一个阶段。 #### 方案二:利用 `Promise.all` 或者 `Promise.allSettled` 实现并行处理 如果我们不在乎各次 API 调用之间是否保持一定的先后顺序的话,也可以借助 ES6 提供的新特性 —— Promises 批量提交请求后同时收集其反馈信息: ```javascript let requestsPromises = myArray.map(x => doApiCallForX(x)); Promise.all(requestsPromises) .then(responses =>{ responses.forEach(resp => handleResponse(resp)); }).catch(err => globalErrorHandler(err)) .finally(() => notifyUserThatEverythingIsDone()); ``` 这种方法允许我们几乎瞬间启动所有的远程连接(受限于浏览器的最大并发限制),然后一旦全部得到回应就会立刻采取行动。 --- ### 总结 尽量避免把 `async/await` 放进非原生支持异步入口点(像普通的 `.map()`,`.filter()`,以及本题提到的 `.forEach()`)之中;而是应当寻找合适的结构来适配我们的实际应用场景需求。比如对于需要维持清晰有序性的场合来说,传统的显式声明型循环就是理想之选;而对于追求高效快速完成多项独立作业的情形下,则应优先考虑采用基于承诺链表的设计思路。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值