在 Vue3 中封装的 Axios 实例中,若需要为部分接口提供手动取消请求的功能


核心思路

  1. 封装接口时返回 Promiseabort 方法
    为需要支持取消的接口返回一个对象,包含 promiseabort 方法,用户可通过 abort 主动中断请求。
  2. 使用 AbortControllerCancelToken
    • 推荐 AbortController(浏览器原生 API,兼容性更好)。
    • 若使用 axiosCancelToken,需额外处理。

实现步骤

1. 封装 Axios 实例
javascript 复制代码
// src/api/index.js
import axios from 'axios';

// 创建 Axios 实例
const instance = axios.create({
  baseURL: '/api',
  timeout: 10000,
});

// 封装需要支持取消的接口
export const api = {
  // 需要支持取消的接口
  getUsers() {
    const controller = new AbortController();
    const { signal } = controller;

    const promise = instance.get('/users', { signal });

    return {
      promise,
      abort: () => controller.abort(), // 提供 abort 方法
    };
  },
  
  // 不需要取消的接口
  getPosts() {
    return instance.get('/posts');
  },
};

2. 在组件中使用
js 复制代码
<template>
  <div>
    <!-- 组件内容 -->
  </div>
</template>

<script setup>
import { onBeforeUnmount } from 'vue';
import { api } from '@/api';

// 调用支持取消的接口
const { promise: usersPromise, abort: abortUsers } = api.getUsers();

// 处理数据
usersPromise
  .then((response) => {
    console.log('用户数据:', response.data);
  })
  .catch((error) => {
    if (error.name === 'AbortError') {
      console.log('用户请求被取消');
    } else {
      console.error('请求失败:', error);
    }
  });

// 在组件卸载时取消请求
onBeforeUnmount(() => {
  abortUsers(); // 主动取消请求
});
</script>

关键点说明

  1. 返回 Promiseabort 方法

    • 接口函数返回一个对象,包含 promiseabort 方法。
    • 用户可通过 abort 方法随时中断请求。
  2. 错误处理

    • 捕获 AbortError 判断是否为用户主动取消。
  3. 生命周期管理

    • onBeforeUnmount 中调用 abort,确保组件卸载时取消未完成的请求。

扩展场景

场景 1:手动触发取消
javascript 复制代码
// 在用户点击取消按钮时触发
const cancelButton = () => {
  abortUsers(); // 主动调用 abort 方法
};
场景 2:防抖/节流中的取消
javascript 复制代码
// 输入搜索框时,取消之前的请求
let currentAbortFn = null;

const handleSearch = (query) => {
  // 取消之前的请求
  if (currentAbortFn) currentAbortFn();

  const { promise, abort } = api.getUsers({ query });
  currentAbortFn = abort;

  promise.then((data) => {
    // 处理数据
  });
};

其他注意事项

  1. 服务端支持

    • 前端取消请求仅终止浏览器端处理,服务端可能仍在执行 。若需服务端终止操作,需传递唯一标识(如 requestId)。
  2. 避免重复调用

    • 确保 abort 方法在适当的时候调用(如组件卸载或用户主动操作),避免多次调用。

完整封装示例

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

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

export const api = {
  // 需要支持取消的接口
  getUsers(params) {
    const controller = new AbortController();
    const { signal } = controller;

    const promise = instance.get('/users', {
      params,
      signal,
    });

    return {
      promise,
      abort: () => controller.abort(),
    };
  },
  
  // 不需要取消的接口
  getPosts() {
    return instance.get('/posts');
  },
};

总结

通过以下步骤实现手动取消接口:

  1. 在封装接口时,返回 Promiseabort 方法。
  2. 在组件中通过 onBeforeUnmount 或用户操作调用 abort
  3. 错误处理时区分取消错误与其他错误。

此方法既保持了接口的灵活性,又实现了对特定请求的精准控制。

相关推荐
jvxiao1 小时前
你真的懂作用域吗?从编译原理角度深度 JS 的作用域
前端·javascript
Darling噜啦啦1 小时前
二叉树与递归算法实战:从树结构到 LeetCode 爬楼梯,一文吃透前端数据结构与递归思维
前端·javascript·数据结构
xsbcme2 小时前
VueTabRouter 插件实践(一):多标签页不是一排 TabBar
vue.js
Sammyyyyy2 小时前
月之暗面 Kimi Code 0.4.0 发布,终端 AI 编码助手全面采用 TypeScript,实现毫秒级启动
前端·javascript·人工智能·ai·typescript·servbay
宋拾壹2 小时前
fastadmin列表中查看列表,并且添加增加相应的数据
javascript·php·fastadmin
云水一下3 小时前
Vue.js从零到精通系列(三):组件化基础——Props、Emits、插槽与生命周期
前端·javascript·vue.js
小糯米6014 小时前
JavaScript表达式与运算符
开发语言·javascript·ecmascript
体验家4 小时前
体验家 XMPlus 网页端问卷 SDK 技术解析:用几行 JavaScript 实现精准场景触发与防打扰机制
开发语言·前端·javascript
VidDown4 小时前
VidDown 工具站:视频分辨率技术
javascript·网络·编辑器·音视频·视频编解码·视频
小鹿软件办公5 小时前
倒计时开启:Chromium 宣布几周内将全面切断 MV2 扩展支持
开发语言·javascript·ublock origin