在umijs中,使用AbortController取消已经发送的请求(打断请求),可以解决重复请求,根据需要取消指定的Web请求

背景介绍

AbortController是一个控制器对象,可以通过new AbortController()的方式生成控制器实例对象,该控制器实例对象可以根据需要取消/打断一个或多个指定的Web请求

问题描述

多次请求同一个接口(参数不同),数据量大的情况下接口返回时间较长,数据量小接口返回时间短,多次请求不同数据量的同一个请求就会出现最后返回的接口不是我当前最后选择的接口,而是数据量最大的接口,当我们请求新的接口时,需要取消还没有返回数据的接口

解决方案

我们通过将发送的所有请求存进队列(用数组结构)中,发送请求的时候入队(push),请求响应的时候出队(shift),标记所有请求。在每次发送请求的时候先判断当前请求url在请求队列中是否重复,如果重复,将队列中重复请求取消,保留当前请求。

代码实现

以下代码全写在请求配置文件中

创建一个队列:

interface obj {
  Time: Number,
  Url: string,
  Params: { [key: string]: any },
  controller: any,
  signal: any,
}
const queue: obj[] = []; //请求队列

请求拦截器方法:

/** 请求拦截器 */
const commonProcRequest = (url: string, options: RequestOptionsInit) => {
  // 每次都会创建一个新的AbortController实例,每次创建的都不一样
  const controller = new AbortController();
  const { signal } = controller;
  /** 监听打断成功后的回调 */
  signal.addEventListener('abort', () => {
    console.log('aborted!');
  });
  if (options.headers) {
    options.headers['token'] = localStorage.getItem('token');
  }
  
  console.log('queue',queue);
  /** 如果请求队列中有相同请求并且请求为获取设备,则取消请求 */
  if(queue.length >= 1 && url === '/sys/sysResource/getDeviceAll'){
    queue.forEach(item => {
      if(item.Url === '/sys/sysResource/getDeviceAll'){
        console.log('打断了创建时间为',item.Time,"的请求");
        /** 取消请求的方法 controller.abort();  */
        item.controller.abort(); 
        
      }
    })
  }
  queue.push({
    Time: new Date().getTime(),
    Url: url,
    Params: options.data,
    controller: controller, //每次创建的abortcontroller实例都不一样
    signal: signal,
  }) //入队
  
  return {
    url: url,
    options: {
      ...options,
      signal, // 将AbortSignal传给options
    },
  };
};

在请求响应处出队

responseInterceptors: [
    async (response: Response) => {
      queue.shift();
      ...
      return response;
    },
  ],

如果项目监听了请求错误,需要在错误处理方法中添加代码,屏蔽取消请求的报错:

if (res.name === 'AbortError') {
    return Promise.resolve(res);
}

  • 13
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值