前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >你可能忽略的 async/await 问题

你可能忽略的 async/await 问题

作者头像
savokiss
发布2020-02-26 11:40:46
5400
发布2020-02-26 11:40:46
举报
文章被收录于专栏:码力全开码力全开
async/await 大家肯定都用过,在处理异步操作的时候真的是很方便。

如果有还不熟悉的小伙伴可以看笔者之前的文章:

带你理解 async/await

那今天主要讲一些在使用 async/await 时容易忽略和犯错的地方。

一个例子

下面是一个常见的 Vue 代码片段:

代码语言:javascript
复制
 async initStore(query) {
    await this.getConfig();
    await this.getUser();
    await this.checkRussianContext(query);

    await this.getBasket(this.$store.state.config.selectedCurrency),

    await this.$store.dispatch('options/fetchOptions', {
      basket : this.$store.state.basket,
    });
 },

上面的代码中,每一行都会 等待上一行的结果返回后才会执行。比如其中的 getUser 会等待 getConfig 数据返回之后才会执行。

当看到这段代码时,脑子里应该注意到以下几点:

  • 如果某一行的执行不需要上面一行的数据怎么办?为什么阻塞代码使整个应用程序变慢呢?
  • 我们可不可以让所有不相关的方法一起执行?比如使用 Promise.all
  • 能否让相关的方法使用 then 进行链式调用来避免阻塞剩余代码?

本篇文章的重点就是通过分析 async/await 可能引发的问题,帮你找到 代码的坏味道

无关的请求请并行执行

让我们来看一些具体的数据下的情况。

下面是示例代码:

代码语言:javascript
复制
const getUserData = async () => {
  // 获取一张随机的狗狗图片作为头像
  const res = await fetch('https://dog.ceo/api/breeds/image/random')
  const { message } = await res.json()

  // 获取随机生成的用户信息
  const user = await fetch('https://randomuser.me/api/')
  const { results } = await user.json()

  // ...
}

上面的代码在 fast 3G (使用 Chrome 开发者工具模拟)下执行 100 次,平均执行时间为 1231.10ms

但是很显然,第二个请求并不需要第一个请求的结果,所以我们修改成以下代码并执行 100 次:

代码语言:javascript
复制
const getUserDataFaster = async () => {
  // 两个请求并行执行
  const [res, user] = await Promise.all([
    fetch('https://dog.ceo/api/breeds/image/random'),
    fetch('https://randomuser.me/api/')
  ])
  const [{ message }, { results }] = await Promise.all([res.json(), user.json()])

  // ...
}

我们得到的平均执行时间为 612.50ms,几乎节省了一半时间。

划重点:尽可能地把查询请求并行执行。

无关的代码你不必等

再来例子:

代码语言:javascript
复制
async initStore(query) {
   await Promise.all([
     this.getConfig(),
     this.getUser(),
     this.checkRussianContext(query)
   ])

   await this.getBasket(this.$store.state.config.selectedCurrency),

   await this.$store.dispatch('options/fetchOptions', {
     basket : this.$store.state.basket,
   });

   await initBooking()
},

前面的 3 个请求是并行执行的,而下一段代码依赖了前面获取的数据,所以需要在其后执行,但是你有没有发现其中的问题?

initBooking 这个小可怜只能等到 getBasketfetchOptions 完成之后才能执行,尽管它不需要这两个方法的任何数据。

一个简单的解决办法是将 await 换成 .then 来使用:

关于这个用法可以看开头的另一篇文章

代码语言:javascript
复制
async initStore(query) {
  await Promise.all([
    this.getConfig(),
    this.getUser(),
    this.checkRussianContext(query)
  ])

  this.getBasket(this.$store.state.config.selectedCurrency).then(async () => {
    await this.$store.dispatch('options/fetchOptions', {
      basket : this.$store.state.basket,
    });
  })

  await initBooking()
},

这样的话,getBasketinitBooking 都可以并行执行了。

结论

async/await 是 JavaScript 中的一个非常棒的特性,我们在享受它们便利写法的同时,也要清楚它们可能引发的问题。有机会审查一下你的代码,看有没有可以并行运行的代码块吧~

Thanks for reading~

参考链接

  • The dangers of async/await
本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2020-02-14,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 码力全开 微信公众号,前往查看

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

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一个例子
  • 无关的请求请并行执行
  • 无关的代码你不必等
  • 结论
  • 参考链接
相关产品与服务
云开发 CLI 工具
云开发 CLI 工具(Cloudbase CLI Devtools,CCLID)是云开发官方指定的 CLI 工具,可以帮助开发者快速构建 Serverless 应用。CLI 工具提供能力包括文件储存的管理、云函数的部署、模板项目的创建、HTTP Service、静态网站托管等,您可以专注于编码,无需在平台中切换各类配置。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档