js - 取消接口请求

js - 取消接口请求

axios - cancelToken

参考资料

axios
axios 之cancelToken原理以及使用
axios取消接口请求
axios中断请求cancelToken

use

CancelToken.source

可以使用 CancelToken.source 工厂方法创建 cancel token,像这样:

const CancelToken = axios.CancelToken;
const source = CancelToken.source();
axios.get('/user/12345', {//get请求在第二个参数
cancelToken: source.token
}).catch(function(thrown) {
if (axios.isCancel(thrown)) {
console.log('Request canceled', thrown.message);
} else {
// 处理错误
}
});
axios.post('/user/12345', {
name: 'new name'
}, {
cancelToken: source.token//post请求在第三个参数
})
// 取消请求(message 参数是可选的)
source.cancel('Operation canceled by the user.');

构造函数

还可以通过传递一个 executor 函数到 CancelToken 的构造函数来创建 cancel token:

const CancelToken = axios.CancelToken;
let cancel;
axios.get('/user/12345', {
cancelToken: new CancelToken(function executor(c) {
// executor 函数接收一个 cancel 函数作为参数
cancel = c;
})
});
// cancel the request
cancel();

notice

  1. 可以使用同一个 cancel token 取消多个请求
  2. source.cancel函数执行后,会永久取消请求(如何取消拦截?)
const CancelToken = axios.CancelToken;
const source = CancelToken.source();
axios.get('/user/12345', {//get请求在第二个参数
cancelToken: source.token // 1. 使用同一个 cancel token 取消多个请求
}).catch(function(thrown) {
if (axios.isCancel(thrown)) {
console.log('Request canceled', thrown.message);
} else {
// 处理错误
}
});
axios.post('/user/12345', {
name: 'new name'
}, { //post请求在第三个参数
cancelToken: source.token // 1. 使用同一个 cancel token 取消多个请求
})
// 取消请求(message 参数是可选的)
source.cancel('Operation canceled by the user.'); // 2. source.cancel函数执行后,会永久取消请求

如何取消拦截

使用构造函数创建 cancel token,每次请求使用不同的 cancel token,请求前执行前一个cancel token

eg:重复请求只执行最后一次

<template>
<div>
<el-button @click="login">login</el-button>
<el-button @click="forLogin">forLogin</el-button>
</div>
</template>
<script>
import axios from 'axios';
const CancelToken = axios.CancelToken;
import { getToken } from '@/utils/cookies'; // get token from cookie
export default {
async mounted() {},
methods: {
CancelToken() {},
forLogin() {
for (let i in Array(5).fill('')) {
this.login();
}
},
async login() {
try {
this.cancelToken();
const res = await this.$Request({
url: `${
this.$store.state.config.loginCenterBaseUrl
}/api/v2/user/${getToken()}`,
// cancelToken: source.token,
cancelToken: new CancelToken((cancel) => {
this.cancelToken = cancel;
}),
});
this.cancelToken = () => {};
console.log('sucess:', res);
} catch (err) {
if (axios.isCancel(err)) {
console.error('Request canceled', err.message);
} else {
// 处理错误
console.error(err);
}
}
},
},
};
</script>

重复点击问题***

开发的时候会遇到一个重复点击的问题,短时间内多次点击同一个按钮发送请求会加重服务器的负担,消耗浏览器的性能,多以绝大多数的时候我们需要做一个取消重复点击的操作

axios拦截器

import axios from 'axios';
axios.defaults.timeout = 5000;
axios.defaults.baseURL ='';
let pending = []; //声明一个数组用于存储每个ajax请求的取消函数和ajax标识
let cancelToken = axios.CancelToken;
let removePending = (ever) => {
for(let p in pending){
if(pending[p].u === ever.url + '&' + ever.method) { //当当前请求在数组中存在时执行函数体
pending[p].f(); //执行取消操作
pending.splice(p, 1); //把这条记录从数组中移除
}
}
}
//http request 拦截器
axios.interceptors.request.use(
config => {
config.data = JSON.stringify(config.data);
config.headers = {
'Content-Type':'application/x-www-form-urlencoded'
}
// ------------------------------------------------------------------------------------
removePending(config); //在一个ajax发送前执行一下取消操作
config.cancelToken = new cancelToken((c)=>{
// 这里的ajax标识我是用请求地址&请求方式拼接的字符串,当然你可以选择其他的一些方式
pending.push({ u: config.url + '&' + config.method, f: c });
});
// -----------------------------------------------------------------------------------------
return config;
},
error => {
return Promise.reject(err);
}
);
//http response 拦截器
axios.interceptors.response.use(
response => {
// ------------------------------------------------------------------------------------------
removePending(res.config); //在一个ajax响应后再执行一下取消操作,把已经完成的请求从pending中移除
// -------------------------------------------------------------------------------------------
if(response.data.errCode ==2){
router.push({
path:"/login",
querry:{redirect:router.currentRoute.fullPath}//从哪个页面跳转
})
}
return response;
},
error => {
return Promise.reject(error)
}
)

原生js - abort()

<body>
<div class="page" id="app">
<button class="get-msg">获取数据</button>
<button class="cancel">取消获取</button>
</div>
<script>
var currentAjax = null;
$('.get-msg').click(function () {
currentAjax = $.ajax({
type: 'GET',
url: 'http://jsonplaceholder.typicode.com/comments',
success: function (res) {
console.log(res);
},
error: function (err) {
console.log('获取失败');
},
});
});
$('.cancel').click(function () {
if (currentAjax) {
currentAjax.abort();
}
});
</script>
</body>
posted @   zc-lee  阅读(930)  评论(0编辑  收藏  举报
点击右上角即可分享
微信分享提示