关于使用Promise/async await解决循环请求接口的问题

更新日期: 2020-02-03 阅读: 3.5k 标签: 接口

问题背景:

我有一组list,包含了所有的预定id,现在我需要循环这组id去请求一个接口,以获取详情信息.
这里需要注意的点是:我需要的信息并不是一个接口可以请求完的,而是需要循环请求接口

那么,我如何确保这些接口全部请求完了才执行下面的操作呢?

list.map(item => {
     axios.get(url).then(data => {
        this.$set(item,"list",data.data.data)
    });
});
list.map((item)=>{
    console.log(item.list) // undefined
})

我们可以发现,如果不进行任何包装,仅仅是这样一个上下结构的逻辑,我们输出新插入的值时,发现全部都是undefined,因为请求是异步的.


解决方法:

使用Promise解决
let queue = list.map(item => {
        return new Promise(resolve => {
          axios.get(url).then(data => {
              resolve(data.data.data);
           });
       });
});
Promise.all(queue).then(result => {
     //TODO
     //执行后续操作
});

Promise.all 方法会等待所有的 异步任务执行结束了,然后将所有任务的执行结果都统一的放到一个数组中,然后传给自己的 then

另外,Promise.all是异步请求并行操作
我们可以通过下面的代码证实一下

console.time("test")
var list = [1,2,3]
var queue =list.map((item)=>{
    return new Promise((resolve,rejcet)=>{
        setTimeout(() => {
           resolve(item)
        }, 1000);
    })
})
Promise.all(queue).then((res)=>{
    console.log(res)
})

console.timeEnd("test")

image.png
我们可以发现整个程序用了1s就运行完了,而非3s

如果你在使用Promise.all时出现了以下报错:
cannot read property Symbol(Symbol.iterator)
那么请注意Promise.all的正确写法是:
Promise.all([promise1,promise2,promise3])
而不是:Promise.all(promise1,promise2,promise3)

使用async/await结合Promise (串行)
async getlist() {
  for (const item of list) {
    const column = await getlistdetail(item)
  }
  //TODO
  //后续操作
},
getlistdetail(item) {
      return new Promise(resolve => {
        axios
          .get(url).then(data => {
            this.$set(item, "roundlist", data.data.data);
            resolve(data.data.data);
          });
      });
    }

由于异步等待axios返回的本身就是一个Promise对象,所以重新再封装一层的必要性不大,可以直接return一个Promise

 async getlistdetail(item) {
      await axios.get(url)
        .then(data => {
          this.$set(item, "roundlist", data.data.data);
          return data.data.data;
        });
    },
使用async/await结合Promise (串行)

最佳解决方法 该方法代码可读性比较高

async getlist() {
      let list = await this.getlist();//获取id合集
      const promises = list.map(x => this.getlistdetail(x));//根据每个id分别请求接口获取详情
      for (const promise of promises) {
        const column = await promise;
      }
      //TODO
      //后续操作
}

来自:https://segmentfault.com/a/1190000022224202

本文内容仅供个人学习、研究或参考使用,不构成任何形式的决策建议、专业指导或法律依据。未经授权,禁止任何单位或个人以商业售卖、虚假宣传、侵权传播等非学习研究目的使用本文内容。如需分享或转载,请保留原文来源信息,不得篡改、删减内容或侵犯相关权益。感谢您的理解与支持!

链接: https://fly63.com/article/detial/8484

相关推荐

常用HTTP接口测试工具对比

从功能上Jmeter最为强大,可以测试各种类型的接口,不支持的也可以通过网上或自己编写的插件进行扩展。SoapUI专门针对HTTP类型的两种接口,其初衷更是专门测试Soap类型接口,对于其他协议的接口不支持

免费的公共API接口_WebService接口大全

这篇文章为大家整理一下免费,常用的的WebService接口,列举一些搜集到的免费的公共API接口,希望对你有所帮助,天气预报Web服务,数据来源于中国气象局;IP地址来源搜索 WEB 服务;随机英文、数字和中文简体字

前后端分离项目的跨域及保持Session会话

当Web项目前后端分离开发的时候, 由于域名不一致, 会出现无法请求和无法维持会话的情况,在前端Ajax请求后台的时候, 打开控制台可以看到, 每一次请求之前都会有一次OPTIONS类型的请求

JSON API免费接口

各种提供JSON格式数据返回服务网站的API接口,为大家搜集了一些能够返回JSON格式的服务接口。部分需要用JSONP调用。

vue中使用proxy配置不同端口和ip接口

使用vue-cli创建的项目,开发地址是localhost:8080,由于后台开发不同的模块,导致每个模块请求的ip和端口号不一致,解决问题:在vue.config.js中配置不同的端口号

前后端分离,接口对接问题

关于前后端对接问题下面好程序员web前端学习路线为大家解答。对于前后端分离开发,必须要写接口文档。否则,第一:前后端开发没有标准,没有依据。第二:容易扯皮,没法追踪,职责不清

TypeScript接口(Interfaces)来定义对象的类型

在 TypeScript 中,我们使用接口(Interfaces)来定义对象的类型。在面向对象语言中,接口(Interfaces)是一个很重要的概念,它是对行为的抽象,而具体如何行动需要由类(classes)去实现(implements)

vue 项目接口管理

在vue开发中,会涉及到很多接口的处理,当项目足够大时,就需要定义规范统一的接口,如何定义呢?方法可能不只一种,本文使用axios+async/await进行接口的统一管理。

vue项目接入mock&& axios 通用配置

兵马未动,粮草先行; 同理,项目开发过程中经常会出现接口未出, 前端页面已搭建完毕的情况;此时为了提高前端的开发效率,解放生产力,我们 FE 可以按照预定的接口文档做一些接口模拟的工作

Mock.js模拟接口数据

现在开发已经是前后端分离了,前端和后端可以同时进行开发,互不影响,但是有些时候后端开发的接口慢于前端,导致前端需要等待后端的接口完成才能完成前后端对接,为了解决这个痛点,出现了模拟接口数据的方案

点击更多...

内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!