前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >JS循环中使用async、await的正确姿势

JS循环中使用async、await的正确姿势

作者头像
科技新语
发布2022-12-15 16:10:04
3.5K0
发布2022-12-15 16:10:04
举报

概览(循环方式 - 常用)

  • for
  • map
  • forEach
  • filter

声明遍历的数组和异步方法

声明一个数组:⬇️

代码语言:javascript
复制
const skills = ['js', 'vue', 'node', 'react']

再声明一个promise的异步代码: ⬇️

代码语言:javascript
复制
function getSkillPromise (value) {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      resolve(value)
    }, 1000)
  })
}

for 循环中使用

由于for循环并非函数,而asyncawait需要在函数中使用,因此需要在for循环外套一层function

代码语言:javascript
复制
async function test () {
  for (let i = 0; i < skills.length; i++) {
    const skill = skills[i]
    const res = await getSkillPromise(skill)
    console.log(res)
  }
}

test() // 调用
for.gif
for.gif

当使用await时,希望JavaScript暂停执行,直到等待 promise 返回处理结果。上述结果意味着for循环中有异步代码,是可以等到for循环中异步代码完全跑完之后再执行for循环后面的代码。 但是他不能处理回调的循环,如forEachmapfilter等,下面具体分析。

map 中使用

map中使用await, map 的返回值始是promise数组,这是因为异步函数总是返回promise

代码语言:javascript
复制
async function test () {
  console.log('start')
  const res = skills.map(async item => {
    return await getSkillPromise(item)
  })
  console.log(res)
  console.log('end')
}

test()

结果:始终为promise数组

代码语言:javascript
复制
start
[
  Promise { <pending> },
  Promise { <pending> },
  Promise { <pending> },
  Promise { <pending> }
]
end

若果你想要等到promise的返回结果,可以使用promise.all()处理一下

代码语言:javascript
复制
async function test () {
  console.log('start')
  const res = skills.map(async item => {
    return await getSkillPromise(item)
  })
  const resPromise = await Promise.all(res)
  console.log(resPromise)
  console.log('end')
}

test()

// 结果
start
[ 'js', 'vue', 'node', 'react' ]
end

forEach 中使用

先上代码和结果

代码语言:javascript
复制
async function test () {
  console.log('start')
  skills.forEach(async item => {
    const res = await getSkillPromise(item)
    console.log(res)
  })
  console.log('end')
}

test()

预期结果

代码语言:javascript
复制
'Start'
'js'
'vue'
'node'
'react'
'End'

实际结果 在forEach循环等待异步结果返回之前就执行了console.log('end')

代码语言:javascript
复制
'Start'
'End'
'js'
'vue'
'node'
'react'

JavaScript 中的 forEach不支持 promise 感知,也支持 asyncawait,所以不能在 forEach 使用 await

filter 中使用

使用filter过滤itemvue或者react的选项

正常使用 filter

代码语言:javascript
复制
async function test () {
  console.log('start')
  const res = skills.filter(item => {
    return ['vue', 'react'].includes(item)
  })
  console.log(res)
  console.log('end')
}

test() // 调用

// 结果
start
[ 'vue', 'react' ]
end

使用 await后:

代码语言:javascript
复制
async function test () {
  console.log('start')
  const res = skills.filter(async item => {
    const skill = await getSkillPromise(item)
    return ['vue', 'react'].includes(item)
  })
  console.log(res)
  console.log('end')
}

test()

预期结果:

代码语言:javascript
复制
start
[ 'vue', 'react' ]
end

实际结果:

代码语言:javascript
复制
[ 'js', 'vue', 'node', 'react' ]
end

结论:因为异步函数getSkillPromise返回结果返回的promise总是真的,所以所有选项都通过了过滤

本文系转载,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文系转载前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 概览(循环方式 - 常用)
  • 声明遍历的数组和异步方法
  • for 循环中使用
  • map 中使用
  • forEach 中使用
  • filter 中使用
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档