【Vue】取消重复请求

在开发过程中经常会遇到这样的问题:根据条件查询列表数据,当数据量过大时,查询速度会有延迟,此时如果变更一个查询范围更小条件,此时第二次发起的请求延迟较低,数据就会先返回到前端页面,可能过了几秒钟之后,第一次请求的数据才会在页面上渲染出来,这样的话旧数据就覆盖了新的数据,我们页面上展示的就是不符合预期的脏数据了。

此时,解决方法有两种。

第一种,在前端页面定义一个全局 uuid,请求的时候发送给后端,后端接口执行完毕之后将该 uuid 返回给前端,前端在处理逻辑的时候去校验当前的 uuid 与后端返回回来的 uuid 是否相同,是则执行对应的逻辑。

实现步骤:

  • 前端生成一个全局唯一的 uuid
  • 后端接收该 uuid 并返回给前端
  • 前端校验该 uuid 是否是最新的 uuid

第二种,由于我前端使用的是 vue,在 vue 中有一种“取消请求”的技术。当重复发起同一个请求之后,vue 会取消之前的所有请求,保留最新的请求。需要注意的是,vue 只作用于前端而无法取消后端程序的执行。例如连续发起 3 次请求,前面两次都是高延迟请求,因此阻塞在后端接口,第三次响应较快,此时前端就会断开前面两次前后端的连接(实际上是逐个断开的,请求 2 断开了请求 1,请求 3 断开了请求 2)。

实现步骤:

  • 用户触发一个新的异步请求时,前端生成一个取消令牌(CancelToken)
  • 如果之前有类似的请求在进行中,前端会使用取消令牌取消之前的请求。
  • 新的请求会使用新生成的取消令牌
  • 如果前一个请求被取消,后端可以根据需要处理这个取消请求。

我采用两者结合的方式,具体实现如下:

1、导入 axios

import axios from "axios"

2、定义一个全局 uuid 和存储取消令牌的字段

export default {
	data() {
		uuid: '',		// 全局uuid
		cancelToken: axios.CancelToken.source(),    // 取消令牌
	}
}

3、具体执行逻辑

// 获取数据列表
fetchData() {
this.uuid = Math.random().toString(36).slice(-8)  // 生成8位随字符
    // 取消上一个请求
    this.cancelToken.cancel('取消上一个请求')
    // 创建新的取消令牌
    this.cancelToken = axios.CancelToken.source()
    // 发起新的请求
    axios.post(, this.listQuery, {
    headers: {
        // ...一些头部信息,例如token	
        token: yourToken,
    },
    cancelToken: this.cancelToken.token
    }).then(resp => {
        let oldUuid = resp.uuid
        if (oldUuid === this.uuid) {		// 校验uuid
        	// ...代码逻辑
        } else {
            // ...代码逻辑
        }
	}).catch(error => {
        if (axios.isCancel(error)) {
        	console.log('请求被取消', error.message)
        } else {
        	console.log('请求错误', error)
        }
    })
},

由于后端接口操作比较简单,这里就不展示后端接口了。

希望能帮到你!

Vue项目中,我们通常会使用axios来发送请求。为了方便管理和维护代码,我们可以将axios请求进行统一封装。 首先,我们需要在项目中安装axios: ``` npm install axios --save ``` 然后,我们可以在src目录下新建一个api目录,用来存放所有的接口请求。在api目录下,我们可以新建一个request.js文件,用来封装axios请求: ``` import axios from 'axios'; // 创建axios实例 const service = axios.create({ baseURL: process.env.BASE_API, // api的base_url timeout: 5000 // 请求超时时间 }); // request拦截器 service.interceptors.request.use(config => { // 在请求发送之前做一些处理 return config; }, error => { // 处理请求错误 console.log(error); Promise.reject(error); }) // respone拦截器 service.interceptors.response.use( response => { // 在接收响应做一些处理,例如统一的错误处理 return response.data; }, error => { console.log('err' + error);// for debug return Promise.reject(error); } ) export default service; ``` 这里我们创建了一个axios实例,并进行了一些配置,例如设置请求超时时间、拦截请求和响应等。并且将其导出,其他地方可以直接引用该实例进行请求。 接下来,我们可以在api目录下新建一个模块,例如user.js,用来存放用户相关的接口请求。在user.js中,我们可以这样写: ``` import request from '@/api/request'; export function login(username, password) { return request({ url: '/user/login', method: 'post', data: { username, password } }) } export function getInfo(token) { return request({ url: '/user/info', method: 'get', params: { token } }) } ``` 这里我们引入了之前封装的axios实例,然后对外导出两个方法login和getInfo,分别用来请求登录和获取用户信息的接口。 最后,我们可以在Vue组件中使用这些接口,例如: ``` import { login, getInfo } from '@/api/user'; export default { name: 'Login', methods: { handleLogin() { login(this.username, this.password).then(response => { // 处理登录成功的逻辑 getInfo(response.token).then(info => { // 处理获取用户信息成功的逻辑 }) }) } } } ``` 这里我们引入了之前定义的login和getInfo方法,并在handleLogin方法中调用它们。需要注意的是,我们在处理登录成功后,还调用了getInfo方法来获取用户信息。这里我们可以看到,我们可以将多个请求串联起来进行处理。 在实际开发中,我们经常会遇到重复请求的问题,例如多次点击提交按钮或者页面切换时进行数据加载等。为了避免重复请求,我们可以对axios进行进一步封装,例如: ``` import axios from 'axios'; // 定义一个空的数组,用来存储每个请求取消函数和标识 const pending = []; const CancelToken = axios.CancelToken; const removePending = (config) => { for (let i = 0; i < pending.length; i++) { if (pending[i].url === config.url + '&' + config.method) { pending[i].cancel(); // 取消重复请求 pending.splice(i, 1); // 删除已经取消请求记录 } } } // 创建axios实例 const service = axios.create({ baseURL: process.env.BASE_API, // api的base_url timeout: 5000 // 请求超时时间 }); // request拦截器 service.interceptors.request.use(config => { // 在请求发送之前做一些处理 removePending(config); config.cancelToken = new CancelToken((c) => { pending.push({ url: config.url + '&' + config.method, cancel: c }); }); return config; }, error => { // 处理请求错误 console.log(error); Promise.reject(error); }) // respone拦截器 service.interceptors.response.use( response => { // 在接收响应做一些处理,例如统一的错误处理 removePending(response.config); return response.data; }, error => { console.log('err' + error);// for debug return Promise.reject(error); } ) export default service; ``` 这里我们定义了一个pending数组,用来存储每个请求取消函数和标识。在每次请求发送之前,我们会先调用removePending函数,用来取消重复请求。在每次请求完成之后,我们会再次调用removePending函数,用来删除已经取消请求记录。这样就可以避免重复请求的问题了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值