详解vue中使用axios对同一个接口连续请求导致返回数据混乱的问题


Posted in Javascript onNovember 06, 2019

业务上出现一个问题:如果连续对同一个接口发出请求,参数不同,有时候先请求的比后请求的返回数据慢,导致数据顺序混乱,或者数据被覆盖的问题,所以需要控制请求的顺序。

解决方法:

1.直接跟后台沟通,将所有参数放到数组里后台统一接收并返回所有数据再由前端进行数据的拆分使用。

2.对于出现返回的数据混乱问题。

假设场景: 页面中需要对三个部门请求对应的部门人员,三个部门人员的数据为一个二维数组,连续发送请求,但由于返回数据的顺序不定,导致数组中的数据顺序不是按照部门的顺序。

解决方法:使用promise.all + axios。

//获取部门人员的请求
getDepartPerson (departData) {
    let that = this
    return new Promise(function(resolve,reject) {
     that.$axios({
       method: 'get',
       url: ...,
       params: {
        ...
       }
      }).then(res => {
       const data = res.data.map(item => {
        return {
         value: item.userId,
         label: item.userName
        }
       })
       resolve(data)
      })
    })
    
   },
   
//使用promise.all控制返回的数据顺序
setPersonData() {
    const data = [{
     departId: 1,
     departName: '部门1'
    }, {
     departId: 2,
     departName: '部门2'
    }, {
     departId: 3,
     departName: '部门3'
    }]
    let promise1 = this.getDepartPerson(data[0])
    let promise2 = this.getDepartPerson(data[1])
    let promise3 = this.getDepartPerson(data[2])
    console.log(promise1,promise2,promise3)
    let that = this
    Promise.all([promise1,promise2,promise3]).then(value => {
     console.log(value) //value返回的数据是按顺序的
    })
   },

这里要注意

在promise中this不能指向vue的,所以在promise使用前赋值

let that = this

3.对于返回数据出现覆盖的问题

假设场景:切换菜单的时候总是会向后台发送同一个请求,不同参数。且假设这几个菜单共用vuex中的一个state,假设从a菜单切换到b菜单中,a返回的数据比b返回的慢,导致覆盖了state,此时虽然切换到b菜单,但是页面上的数据是a的数据。

解决方法:使用axios中的CancelToken,对于之前的请求进行禁止。

//取消接口相同参数不同的处于pending状态下的请求
export const pending = []
let CancelToken = axios.CancelToken
let cancelPending = (config) => {
 for(let i=pending.length-1; i>=0; i--){
  if (!!config) {
   if (pending[i].u === config.url && pending[i].delPending) {
    console.log('delete request')
    pending[i].f() // 取消请求
    pending.splice(i, 1) // 移除当前请求记录
   }
  } else {
   pending[i].f() // 取消请求
   pending.splice(i, 1) // 移除当前请求记录
  }
 }
}

接着在请求前进行拦截

/**
 * 请求前拦截
 */
export function requestSuccessFunc (config) {
 cancelPending(config)
 config.cancelToken = new CancelToken((c) => {
  pending.push({'u': config.url, 'f': c, delPending: config.delPending})
 })
 return config
}

/**
 * 请求结果预处理
 * @param response
 * @returns {Promise<never>}
 */
export function responseSuccessFunc (response) {
 cancelPending(response.config)
}

拓展:如果在切换路由的时候可以将之前页面中请求处于pengding状态的取消

export function routerAfterEachFunc () {
 // 这里可以做路由后操作
 //切换路由时取消之前页面处于pending的请求
 for(let i=pending.length-1; i>=0; i--){
  pending[i].f() // 取消请求
  pending.splice(i, 1) // 移除当前请求记录
 }
 console.log(pending)
}

....

const ROUTER = new Router({
 routes: CONST_ROUTER
})
ROUTER.afterEach(routerAfterEachFunc)
export default ROUTER

4.假设这里不是请求同一个接口,而是上一个接口返回的数据作为下一个接口请求的参数,这是可以使用async await

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
学习ExtJS border布局
Oct 08 Javascript
通用javascript脚本函数库 方便开发
Oct 13 Javascript
无阻塞加载脚本分析[全]
Jan 20 Javascript
jQuery 1.7.2中getAll方法的疑惑分析
May 23 Javascript
jquery()函数的三种语法介绍
Oct 09 Javascript
JS实现隔行换色的表格排序
Mar 27 Javascript
Sublime Text新建.vue模板并高亮(图文教程)
Oct 26 Javascript
Node.js使用MySQL连接池的方法实例
Feb 11 Javascript
详解vue添加删除元素的方法
Jun 30 Javascript
简单说说如何使用vue-router插件的方法
Apr 08 Javascript
基于VUE实现判断设备是PC还是移动端
Jul 03 Javascript
jquery实现简易验证插件封装
Sep 13 jQuery
vuex state中的数组变化监听实例
Nov 06 #Javascript
element的el-table中记录滚动条位置的示例代码
Nov 06 #Javascript
webpack是如何实现模块化加载的方法
Nov 06 #Javascript
node读写Excel操作实例分析
Nov 06 #Javascript
详解vue页面首次加载缓慢原因及解决方案
Nov 06 #Javascript
electron 安装,调试,打包的具体使用
Nov 06 #Javascript
weui中的picker使用js进行动态绑定数据问题
Nov 06 #Javascript
You might like
apache mysql php 源码编译使用方法
2012/05/03 PHP
php ZipArchive压缩函数详解实例
2013/11/06 PHP
php按单词截取字符串的方法
2015/04/07 PHP
PHP实现浏览器中直接输出图片的方法示例
2018/03/14 PHP
Laravel Eloquent ORM 实现查询表中指定的字段
2019/10/17 PHP
JS下拉框内容左右移动效果的具体实现
2013/07/10 Javascript
jQuery学习之prop和attr的区别示例介绍
2013/11/15 Javascript
JQuery中dataGrid设置行的高度示例代码
2014/01/03 Javascript
js之ActiveX控件使用说明 new ActiveXObject()
2014/03/03 Javascript
jQuery.position()方法获取不到值的安全替换方法
2015/03/13 Javascript
Javascript函数式编程语言
2015/10/11 Javascript
jQuery解析返回的xml和json方法详解
2017/01/05 Javascript
Vue.js实现简单动态数据处理
2017/02/13 Javascript
vue与bootstrap实现时间选择器的示例代码
2017/08/26 Javascript
vue生成文件本地打开查看效果的实例
2018/09/06 Javascript
React 使用recharts实现散点地图的示例代码
2018/12/07 Javascript
vue-cli3环境变量与分环境打包的方法示例
2019/02/18 Javascript
JavaScript实现随机点名小程序
2020/10/29 Javascript
[06:16]第十四期-国士无双绝地翻盘之撼地神牛
2014/06/24 DOTA
Python3基础之函数用法
2014/08/13 Python
python实现简单温度转换的方法
2015/03/13 Python
使用Python的PIL模块来进行图片对比
2016/02/18 Python
ActiveMQ:使用Python访问ActiveMQ的方法
2019/01/30 Python
详解如何减少python内存的消耗
2019/08/09 Python
python 画图 图例自由定义方式
2020/04/17 Python
matplotlib制作雷达图报错ValueError的实现
2021/01/05 Python
在C语言中实现抽象数据类型什么方法最好
2014/06/26 面试题
经典安踏广告词
2014/03/21 职场文书
初中升旗仪式演讲稿
2014/05/08 职场文书
红色故事演讲稿
2014/05/22 职场文书
2014年平安夜寄语
2014/12/08 职场文书
go原生库的中bytes.Buffer用法
2021/04/25 Golang
Python面向对象之内置函数相关知识总结
2021/06/24 Python
Python中使用Opencv开发停车位计数器功能
2022/04/04 Python
MySQL 数据表操作
2022/05/04 MySQL
使用opencv-python如何打开USB或者笔记本前置摄像头
2022/06/21 Python