axios取消接口请求

axios取消请求

这里就是分析一下接口请求需要被取消时的一些操作

因为我是用vue写的项目,所以标配用的是axios,怎么在axios中取消已经发送的请求呢?

1.在这之前我们还是先介绍一下原生js的abort()这个方法。

直接上代码会比较好一点

    <divclass="page"id="app"><buttonclass="get-msg">获取数据</button><buttonclass="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>

axios取消接口请求

点击获取数据按钮打印出来的数据

axios取消接口请求

点击获取数据之后~超超超快快快~的手速点击取消获取按钮打印出来的效果

看这两张效果图就知道,我们的abort()方法起作用了!!!!


2.在axios中取消接口请求操作

好了,接下来才是我们的主题,Axios官方提供了一个取消接口请求的方法,但是怎么用这个方法官网写的很简略(不知道是不是我没找全的问题),反正官网的axios取消接口请求累的半死没看懂,后来是扒了很多大佬的博客,才自己理解出来的

Axios 提供了一个 CancelToken的函数,这是一个构造函数,该函数的作用就是用来取消接口请求的,至于怎么用,看代码吧,我在代码中写了注解

    <body><divclass="page"id="app"><button@click="getMsg"class="get-msg">获取数据</button><button@click="cancelGetMsg"class="cancel">取消获取</button><ul><liv-for="item in items">{{item.name}}</li></ul></div><script>
      var app = new Vue({
        el: '#app',
        data: {
          message: 'Hello Vue!',
          items: [],
          cancel: null
        },
        methods: {
          getMsg () {
            let CancelToken = axios.CancelToken
            let self = this
            axios.get('http://jsonplaceholder.typicode.com/comments', {
              cancelToken: new CancelToken(function executor(c) {
                self.cancel = c
                console.log(c)
                // 这个参数 c 就是CancelToken构造函数里面自带的取消请求的函数,这里把该函数当参数用
              })
            }).then(res => {
              this.items = res.data
            }).catch(err => {
              console.log(err)
            })
    
    
            //手速够快就不用写这个定时器了,点击取消获取就可以看到效果了
            setTimeout(function () {
              //只要我们去调用了这个cancel()方法,没有完成请求的接口便会停止请求
              self.cancel()
            }, 100)
          },
          //cancelGetMsg 方法跟上面的setTimeout函数是一样的效果,因为手速不够快,哦不,是因为网速太快,导致我来不及点取消获取按钮,数据就获取成功了
          cancelGetMsg () {
            // 在这里去判断你的id 1 2 3,你默认是展示的tab1,点击的时候不管你上一个请求有没有执行完都去调用这个cancel(),
            this.cancel()
          }
        }
      })
      </script></body>

上两张效果图展示一下:

axios取消接口请求

点击获取数据按钮获获取到了数据

axios取消接口请求

点击获取数据之后,用快快快的佛山无影手点击了取消获取得到的效果

这样,就完美的解决了我遇到的问题了,点击tab切换的时候,网络敢延迟,我就敢掐掉你的请求,保证我下一个请求不被影响


3.重复点击问题

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

在vue开发中,这个方法一样完美解决这一问题,通常我们会封装一遍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)
    }
  )

这是我自己项目用来封装axios的代码,里面加入了取消重复点击事件的方法.

axios取消接口请求

上一篇:axios ajax fetch 区别以及优缺点


下一篇:Create React App脚手架的安装