详解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 相关文章推荐
JQuery自定义事件的应用 JQuery最佳实践
Aug 01 Javascript
定时器(setTimeout/setInterval)调用带参函数失效解决方法
Mar 26 Javascript
Javascript的时间戳和php的时间戳转换注意事项
Apr 12 Javascript
父元素与子iframe相互获取变量和元素对象的具体实现
Oct 15 Javascript
js的2种继承方式详解
Mar 04 Javascript
关于jQuery判断元素是否存在的问题示例探讨
Jul 21 Javascript
JS模拟的Map类实现方法
Jun 17 Javascript
Angularjs为ng-click事件传递参数
Jun 15 Javascript
js CSS3实现卡牌旋转切换效果
Jul 04 Javascript
Angular实现的简单查询天气预报功能示例
Dec 27 Javascript
JavaScript动态添加数据到表单并提交的几种方式
Jun 26 Javascript
JS实现躲避粒子小游戏
Jun 18 Javascript
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
CPU步进是什么意思?i3-9100F B0步进和U0步进区别知识科普
2020/03/17 数码科技
PHP的类 功能齐全的发送邮件类
2006/10/09 PHP
如何实现给定日期的若干天以后的日期
2006/10/09 PHP
php检测文件编码的方法示例
2014/04/25 PHP
Thinkphp实现MySQL读写分离操作示例
2014/06/25 PHP
ThinkPHP使用PHPExcel实现Excel数据导入导出完整实例
2014/07/22 PHP
Zend Framework连接Mysql数据库实例分析
2016/03/19 PHP
Javascript之文件操作
2007/03/07 Javascript
js类的静态属性和实例属性的理解
2009/10/01 Javascript
深入学习JavaScript中的Rest参数和参数默认值
2015/07/28 Javascript
javascript仿百度输入框提示自动下拉补全
2016/01/07 Javascript
JavaScript事件学习小结(五)js中事件类型之鼠标事件
2016/06/09 Javascript
IntersectionObserver API 详解篇
2016/12/11 Javascript
jQuery实现优雅的弹窗效果(6)
2017/02/08 Javascript
vue项目引入字体.ttf的方法
2018/09/28 Javascript
vue实现div拖拽互换位置
2020/07/29 Javascript
详解可以用在VS Code中的正则表达式小技巧
2019/05/14 Javascript
微信小程序左滑删除实现代码实例
2019/09/16 Javascript
python 示例分享---逻辑推理编程解决八皇后
2014/07/20 Python
Python中使用异常处理来判断运行的操作系统平台方法
2015/01/22 Python
Python用GET方法上传文件
2015/03/10 Python
python list元素为tuple时的排序方法
2018/04/18 Python
python批量赋值操作实例
2018/10/22 Python
在python中利用KNN实现对iris进行分类的方法
2018/12/11 Python
详解Python在使用JSON时需要注意的编码问题
2019/12/06 Python
Python实现剪刀石头布小游戏(与电脑对战)
2019/12/31 Python
Jupyter notebook如何实现指定浏览器打开
2020/05/13 Python
基于Python绘制个人足迹地图
2020/06/01 Python
苹果中国官方网站:Apple中国
2016/07/22 全球购物
中国综合网上购物商城:苏宁易购
2016/08/09 全球购物
高中体育教学反思
2014/01/29 职场文书
运动会通讯稿150字
2014/02/15 职场文书
个人遵守党的政治纪律情况对照检查材料思想汇报
2014/09/25 职场文书
追悼会家属答谢词
2015/09/29 职场文书
Nginx优化服务之网页压缩的实现方法
2021/03/31 Servers
Python趣味挑战之实现简易版音乐播放器
2021/05/28 Python