浅析异步循环for await of的使用及执行机制及for of/forEach本质区别和遇到异步时的处理
一、基本介绍
1、语法定义
for await...of
语句创建一个循环,该循环遍历异步可迭代对象以及同步可迭代对象,包括: 内置的 String
, Array
,类似数组对象 (例如 arguments
或 NodeList
),TypedArray
, Map
, Set
和用户定义的异步/同步迭代器。它使用对象的每个不同属性的值调用要执行的语句来调用自定义迭代钩子。
类似于 await
运算符一样,该语句只能在一个async function 内部使用。
2、如何使用
function getTime(seconds){
return new Promise(resolve=>{
setTimeout(() => {
console.log(seconds)
resolve(seconds)
}, seconds);
})
}
function test(){
let arr = [getTime(2000), getTime(300), getTime(1000)]
for (let x of arr){
console.log(x);
}
}
test()
对比区别,这里没有 await 。你可以猜测下这个会怎么输出?分别是 2000的Promise、300的Promise、1000的Promise 顺序输出的,然后再按时间延时打印 300、1000、2000
那么我们再想一下,如果我就是需要按 2000、300、1000 的顺序输出怎么办呢?
首先想到的是 async await 实现是可以的,但是会有缺陷。如果有好几个promise或者异步任务,就会写相应数量的 await,代码量变得庞大臃肿。所以使用 for await of 来实现
此时的输出就是 Promise undefined,然后过 2s 后,按 2000、300、1000 的顺序输出。
3、执行机制:for await of 循环可以暂停循环,当第一个异步执行完成后才会执行下一个,最后结果是让输出结果保持同步顺序输出。
什么意思呢?我们自己直接试试就知道了。
(1)上面代码 arr = [getTime(200), getTime(3000), getTime(1000)] 时,先打印 200,然后过 2s,再一起打印3000、1000
(2)[getTime(1000), getTime(5000), getTime(6000)] 时,先打印 1000,然后过4s,打印 5000,再过1s打印6000
二、async+await 遇见 forEach和 for···of
1、首先看 2 道题,自己先考虑下,能不能答对。
// 定义一个fetch函数模拟异步请求
function fetch(x) {
return new Promise((resolve, reject) => {
console.log('aaa');
setTimeout(() => {
resolve(x)
}, 500 * x)
})
}
// 第一题:
function test() {
let arr = [3, 2, 1]
arr.forEach(async item => {
const res = await fetch(item)
console.log(res)
})
console.log('end')
}
test(); // 输出什么
// 第二题:
async function test() {
let arr = [3, 2, 1]
for (const item of arr) {
const res = await fetch(item)
console.log(res)
}
console.log('end')
}
test(); // 输出什么
第一题:先同时先3个aaa和end,test函数执行完成返回undefined,然后异步回调每隔500ms依次打印1、2、3
第二题:先aaa,1500ms后打印 3,再 aaa,1s后打印2,再aaa,500ms后打印1,再end,再test函数返回值undefined。
这里我理解错了,test函数执行完成,就应该返回 Promise undefined,也就是第二题输出是这样的:
这里需要特别注意下 end 的打印,这里涉及到“协程”的概念理解,asyc await 本质是利用协程来实现的。
2、为什么同样是遍历,输出结果却不一样呢?
因为 for...of 内部处理的机制和 forEach 不同,forEach 是直接调用回调函数,for...of 是通过迭代器的方式去遍历。
3、两者的处理机制:
// 参考下 Polyfill 版本的 forEach,简化后的伪代码:
while (index < arr.length) {
callback(item, index) //我们传入的回调函数
}
而使用迭代器写第二题(既for...of 码的语法糖)等价于:
async function test() {
let arr = [3, 2, 1]
const iterator = arr[Symbol.iterator]() //for of会自动调用遍历器函数
let res = iterator.next()
while (!res.done) {
const value = res.value
const res1 = await fetch(value)
console.log(res1)
res = iterator.next()
}
console.log('end')
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!
2021-03-16 H5唤起APP的解决方案:手机浏览器跳转app内指定页面(URL Scheme使用)
2021-03-16 Android平台、iOS平台设置UrlSchemes,实现被第三方应用调用
2018-03-16 小知识随手记(七)
2018-03-16 浅析API请求慢如何在前端分析以及前端可能存在的原因
2018-03-16 vue3 vite打包部署后访问报错Expected a JavaScript module script but the server responded with a MIME type of