如何封装axios和取消重复请求

1. 创建基础的 axios 实例

首先,创建一个基础的 axios 实例,并设置一些通用的配置,比如基础 URL、超时时间等。

javascript 复制代码
// src/utils/axios.js
import axios from 'axios';

const instance = axios.create({
  baseURL: 'https://api.example.com', // 替换为你的API地址
  timeout: 10000, // 请求超时时间
  headers: {
    'Content-Type': 'application/json',
  },
});

export default instance;

2. 封装请求方法

封装常用的 HTTP 请求方法(如 GET、POST、PUT、DELETE),以便在项目中复用。

javascript 复制代码
// src/utils/request.js
import instance from './axios';

// GET 请求
export const get = (url, params = {}) => {
  return instance.get(url, { params });
};

// POST 请求
export const post = (url, data = {}) => {
  return instance.post(url, data);
};

3. 使用拦截器处理请求和响应

利用 axios 的拦截器功能,可以在请求发送前和响应返回后进行一些全局处理,比如添加请求头token、处理响应错误等。比如状态码统一管理

常见状态码 HTTP状态码:如200(成功)、203(重定向),400(请求错误)、401(未授权)403(授权失败)、404(未找到)、500(服务器错误)等。

javascript 复制代码
// 请求拦截器
instance.interceptors.request.use(
  (config) => {
    // 在发送请求之前做些什么,比如添加token
    const token = localStorage.getItem('token');
    if (token) {
      config.headers.Authorization = `Bearer ${token}`;
    }
    return config;
  },
  (error) => {
    // 对请求错误做些什么
    return Promise.reject(error);
  }
);

// 响应拦截器
instance.interceptors.response.use(
  (response) => {
    // 对响应数据做点什么
    return response.data;
  },
  (error) => {
    // 对响应错误做点什么
    if (error.response) {
      // 服务器响应了状态码, 但不是2xx
      console.error(`Error ${error.response.status}: ${error.response.data}`);
    } else if (error.request) {
      // 请求已发出, 但没有收到响应
      console.error('No response received:', error.request);
    } else {
      // 设置请求时发生了错误
      console.error('Error setting up request:', error.message);
    }
    return Promise.reject(error);
  }
);

4. 取消重复请求

为了避免重复请求,可以使用 axiosCancelToken 功能。以下是一个示例,展示如何在组件卸载时取消请求。

javascript 复制代码
import axios from 'axios';

const instance = axios.create({
  baseURL: 'https://api.example.com',
  timeout: 10000,
});

const pendingRequests = new Map(); // 用于存储正在进行的请求

// 请求拦截器
instance.interceptors.request.use(
  (config) => {
    // 生成请求的唯一标识(可以根据 URL 和参数生成)
    const requestKey = `${config.url}?${new URLSearchParams(config.params).toString()}`;

    // 如果有相同的请求正在执行,取消之前的请求
    if (pendingRequests.has(requestKey)) {
      const cancelToken = pendingRequests.get(requestKey);
      cancelToken();
      pendingRequests.delete(requestKey); // 删除之前的请求
    }

    // 创建新的取消令牌
    const source = axios.CancelToken.source();

    // 存储新的请求
    pendingRequests.set(requestKey, source.cancel);

    // 将取消令牌附加到请求配置中
    config.cancelToken = source.token;

    return config;
  },
  (error) => {
    return Promise.reject(error);
  }
);

// 响应拦截器
instance.interceptors.response.use(
  (response) => {
    // 响应成功后,移除对应的请求
    const requestKey = `${response.config.url}?${new URLSearchParams(response.config.params).toString()}`;
    pendingRequests.delete(requestKey);

    return response;
  },
  (error) => {
    // 响应失败后,移除对应的请求
    const requestKey = `${error.config.url}?${new URLSearchParams(error.config.params).toString()}`;
    pendingRequests.delete(requestKey);

    return Promise.reject(error);
  }
);

export default instance;

代码解释

pendingRequests 的作用

  • 用于存储正在进行的请求的取消令牌(CancelToken)。
  • 键(requestKey)是请求的唯一标识,通常是 URL 和参数的组合。

请求拦截器

  • 在每次请求发送前,检查是否有相同的请求正在执行。
  • 如果有,调用取消函数并移除之前的请求。
  • 创建新的取消令牌并存储到 pendingRequests 中。

响应拦截器

  • 在请求成功或失败后,移除对应的请求记录。

CancelToken 的原理是通过一个 Promise,当调用 cancel 方法时,会触发 Promisereject,从而中断请求。

相关推荐
山楂树の13 分钟前
图像标注大坑:img图片 + Canvas 叠加标注,同步放大后标注位置偏移、对不齐?详解修复方案及亚像素处理原理
前端·css·学习·canva可画
本山德彪15 分钟前
我做了一个拼豆图纸生成器,把照片秒变图纸
前端
DTrader42 分钟前
用TS无法实盘量化? - 实盘均线策略
前端·api
进击的夸父1 小时前
vfojs:Vue 超集架构,外壳React灵魂Vue
前端
编程老船长1 小时前
解决不同项目需要不同 Node.js 版本的问题
前端·vue.js
Wect1 小时前
LeetCode 5. 最长回文子串:DP + 中心扩展
前端·算法·typescript
漫游的渔夫1 小时前
前端开发者做 Agent:别写成一次请求,用 5 步受控循环防止 AI 乱跑
前端·人工智能·typescript
薛定猫AI2 小时前
【深度解析】Gemma Chat 本地 AI 编程 Agent:Electron + MLX + 开源模型的离线 Vibe Coding 实战
javascript·人工智能·electron
kyriewen2 小时前
Webpack vs Vite:一个是“老黄牛”,一个是“猎豹”,你选谁?
前端·webpack·vite
打小就很皮...2 小时前
html2canvas + jsPDF 生成 PDF 的踩坑与解决方案总结
前端·pdf