详解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 相关文章推荐
推荐dojo学习笔记
Mar 24 Javascript
基于jquery的无刷新分页技术
Jun 11 Javascript
图标线性回归斜着移动到指定的位置
Aug 16 Javascript
angular中使用路由和$location切换视图
Jan 23 Javascript
jQuery实现的网页换肤效果示例
Sep 20 Javascript
8 行 Node.js 代码实现代理服务器
Dec 05 Javascript
JS严格模式知识点总结
Feb 27 Javascript
如何从0开始用node写一个自己的命令行程序
Dec 29 Javascript
原生javascript实现连连看游戏
Jan 03 Javascript
使用VueCli3+TypeScript+Vuex一步步构建todoList的方法
Jul 25 Javascript
jQuery zTree树插件的使用教程
Aug 16 jQuery
vue 组件简介
Jul 31 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
PHP PDOStatement对象bindpram()、bindvalue()和bindcolumn之间的区别
2014/11/20 PHP
PHP json_encode() 函数详解及中文乱码问题
2015/11/05 PHP
php使用get_class_methods()函数获取分类的方法
2016/07/20 PHP
PHP利用Cookie设置用户30分钟未操作自动退出功能
2017/07/03 PHP
ThinkPHP框架实现导出excel数据的方法示例【基于PHPExcel】
2018/05/12 PHP
PHP使用标准库spl实现的观察者模式示例
2018/08/04 PHP
Track Image Loading效果代码分析
2007/08/13 Javascript
基于jquery的跨域调用文件
2010/11/19 Javascript
js中array的sort()方法使用介绍
2014/02/20 Javascript
jquery解析XML字符串和XML文件的方法说明
2014/02/21 Javascript
javascript 广告移动特效的实现代码
2016/06/25 Javascript
JS实现评价的星星功能
2017/08/20 Javascript
JS实现li标签的删除
2019/04/12 Javascript
微信小程序获取当前位置和城市名
2019/11/13 Javascript
vue proxy 的优势与使用场景实现
2020/06/15 Javascript
vue实现几秒后跳转新页面代码
2020/09/09 Javascript
Python兔子毒药问题实例分析
2015/03/05 Python
Python常用算法学习基础教程
2017/04/13 Python
对Xpath 获取子标签下所有文本的方法详解
2019/01/02 Python
Django Admin后台添加数据库视图过程解析
2020/04/01 Python
python json.dumps中文乱码问题解决
2020/04/01 Python
python和php学习哪个更有发展
2020/06/17 Python
Python rabbitMQ如何实现生产消费者模式
2020/08/24 Python
python获取时间戳的实现示例(10位和13位)
2020/09/23 Python
VSCODE配置Markdown及Markdown基础语法详解
2021/01/19 Python
python 图像增强算法实现详解
2021/01/24 Python
英国高街品牌:Miss Selfridge(塞尔弗里奇小姐)
2016/09/21 全球购物
财务会计专业应届毕业生求职信
2013/10/18 职场文书
测控技术与通信工程毕业生自荐信范文
2013/12/28 职场文书
学生思想表现的评语
2014/01/30 职场文书
冬季安全检查方案
2014/05/23 职场文书
政府领导干部个人对照检查材料思想汇报
2014/09/24 职场文书
庆祝教师节标语
2014/10/09 职场文书
银行业务授权委托书
2014/10/10 职场文书
Mysql 如何实现多张无关联表查询数据并分页
2021/06/05 MySQL
python APScheduler执行定时任务介绍
2022/04/19 Python