LOGO OA教程 ERP教程 模切知識交流 PMS教程 CRM教程 開發文檔 其他文檔  
 
網站管理員

你在 forEach 里寫的 await,其實根本沒在等!

admin
2025年7月1日 9:36 本文熱度 631

forEach和 async/await的這個組合,就像一對貌合神離的“情侶”,看起來般配,實則互相“背叛”。這個坑,我結結實實地踩過,而且不止一次。

故事的開始:一個看似無害的需求

想象一下,接到一個需求:批量更新一組用戶的狀態。后端提供了一個接口 updateUser(userId),它是一個返回 Promise 的異步函數。第一反應可能就是這樣寫:

const userIds = [1, 2, 3, 4, 5];

async function updateUserStatus(id) {
 console.log(`開始更新用戶 ${id}...`);
 // 模擬一個需要 1 秒的網絡請求
 await new Promise(resolve => setTimeout(resolve, 1000)); 
 console.log(`? 用戶 ${id} 更新成功!`);
 return { success: true };
}

async function batchUpdateUsers(ids) {
 console.log("--- 開始批量更新 ---");

  ids.forEach(async (id) => {
    await updateUserStatus(id);
  });

 console.log("--- 所有用戶更新完畢!---"); // ?? 問題的根源在這里!
}

batchUpdateUsers(userIds);

運行這段代碼,控制臺輸出了什么?不是期望的按順序等待,而是這樣的結果:

看到了嗎?“所有用戶更新完畢!”這句話幾乎是立即打印出來的,它根本沒有“等待”任何 updateUserStatus函數的完成。

問題剖析:forEach到底干了什么?

forEach被設計為同步迭代器。它的工作很簡單:遍歷數組中的每個元素,并為每個元素同步地調用你提供的回調函數。它不關心你的回調函數是同步的還是異步的,也不關心它返回什么。

換句話說,forEach的內心獨白是:

“我的任務就是觸發,觸發,再觸發。至于你傳進來的那個 async函數什么時候執行完?抱歉,那不歸我管,我不會等它的。”


正確的姿勢:如何真正地“等待”?

既然 forEach不行,那我們該用什么?答案是使用那些“懂” Promise 的循環方式。

方案一:老實人 for...of循環(順序執行)

如果我們需要按順序、一個接一個地執行異步操作,for...of循環是你的最佳選擇。它是 async/await的天作之合。

async function batchUpdateUsersInOrder(ids) {
  console.log("--- 開始批量更新 (順序執行) ---");

  for (const id of ids) {
    // 這里的 await 會實實在在地暫停 for 循環的下一次迭代
    await updateUserStatus(id); 
  }

  console.log("--- 所有用戶更新完畢!(這次是真的) ---");
}

運行結果:

這完全符合我們的直覺:等待上一個完成后,再開始下一個。

方案二:效率先鋒 Promise.allmap(并行執行)

在很多場景下,我們并不需要嚴格地按順序執行。這些異步任務之間沒有依賴關系,完全可以并行處理以提高效率。這時,map和 Promise.all的組合就閃亮登場了。

  1. Array.prototype.map:與 forEach不同,map會返回一個新數組。當我們給它一個 async函數時,它會同步地返回一個由 pendingPromise 組成的數組。
  2. Promise.all:這個方法接收一個 Promise 數組,并返回一個新的 Promise。只有當數組中所有的 Promise 都成功完成(resolved)時,這個新的 Promise 才會完成。
async function batchUpdateUsersInParallel(ids) {
 console.log("--- 開始批量更新 (并行執行) ---");

 // 1. map 會立即返回一個 Promise 數組
 const promises = ids.map(id => updateUserStatus(id));

 // 2. Promise.all 會等待所有 promises 完成
 await Promise.all(promises);

 console.log("--- 所有用戶更新完畢!(這次是真的,而且很快) ---");
}

運行結果:

這種方式的總耗時約等于最慢的那個異步任務的耗時,效率極高。

方案三:更靈活的 for...in和傳統 for循環

for...in(用于遍歷對象鍵)和傳統的 for (let i = 0; ...)循環同樣支持 await。它們的工作方式與 for...of類似,都會等待 await的 Promise 完成。

// 傳統 for 循環
for (let i = 0; i < ids.length; i++) {
  await updateUserStatus(ids[i]);
}

為了防止你和我一樣踩坑,這里有一份速記備忘錄:需要按順序執行使用 for...of;需要并行執行,提高效率使用 Promise.allmap,性能最佳,但要注意并發數過高可能帶來的問題;絕對不要用 forEach,它不會等待我們的 await,它只會無情地觸發。


閱讀原文:https://mp.weixin.qq.com/s/QUynwSg3aBjzsNo5WttXDQ


該文章在 2025/7/1 9:36:26 編輯過
關鍵字查詢
相關文章
正在查詢...
點晴ERP是一款針對中小制造業的專業生產管理軟件系統,系統成熟度和易用性得到了國內大量中小企業的青睞。
點晴PMS碼頭管理系統主要針對港口碼頭集裝箱與散貨日常運作、調度、堆場、車隊、財務費用、相關報表等業務管理,結合碼頭的業務特點,圍繞調度、堆場作業而開發的。集技術的先進性、管理的有效性于一體,是物流碼頭及其他港口類企業的高效ERP管理信息系統。
點晴WMS倉儲管理系統提供了貨物產品管理,銷售管理,采購管理,倉儲管理,倉庫管理,保質期管理,貨位管理,庫位管理,生產管理,WMS管理系統,標簽打印,條形碼,二維碼管理,批號管理軟件。
點晴免費OA是一款軟件和通用服務都免費,不限功能、不限時間、不限用戶的免費OA協同辦公管理系統。
Copyright 2010-2025 ClickSun All Rights Reserved

黄频国产免费高清视频,久久不卡精品中文字幕一区,激情五月天AV电影在线观看,欧美国产韩国日本一区二区
亚洲综合中文字幕在线专区 | 中文亚洲欧美丝袜清纯 | 亚洲国产精品久久久久久久 | 亚洲欧美在线观看网站 | 日韩电影免费在线观看视频 | 日韩欧美动漫国产另类中文字幕 |