封装axios

封装axios

网络请求的状态码

封装一个方法用于获取常见状态码对应的中文信息,将他放置在 /src/utils//status.ts:

export const getMessageInfo = (status: number | string): string => {
  let msg = "";
  switch (status) {
    case 400:
      msg = "请求错误(400)";
      break;
    case 403:
      msg = "拒绝访问(403)";
      break;
    case 401:
      msg = "未授权(401)";
      break;
    case 500:
      msg = "服务器错误(500)";
      break;
    case 503:
      msg = "服务不可用(503)";
      break;
    default:
      msg = `连接出错(${status})!`;
  }
  return msg;
};

然后我们在响应拦截器对响应码进行判断提示,如果不是成功响应发出提示给用户,这里直接使用ElementPlus的Message组件,大家可以根据习惯选择其他方式。
/src/utils/request.ts

import { getMessageInfo } from './status';
import { ElMessage } from 'element-plus';

// ...

// axios响应拦截
// 给予用户友好提示
service.interceptors.response.use(
  (response: AxiosResponse) => {
    if (response.status === 200) {
      return response;
    }
    ElMessage({
      message: getMessageInfo(response.status),
      type: 'error',
    });
    return response;
  },
  // 请求失败
  (error: any) => {
    const { response } = error;
    if (response) {
      // 请求已发出,但是不在2xx的范围
      ElMessage({
        message: getMessageInfo(response.status),
        type: 'error',
      });
      return Promise.reject(response.data);
    }
    ElMessage({
      message: '网络异常,请稍后再试!',
      type: 'error',
    });
  }
);
接口管理
  • 在我们成功将axios进项目之后,总是在组件中直接单独引用axios再进行配置请求是十分不方便的,对于一个接口我们可能会有多个组件会用到。

接口一般会有一层最外围的规范
下面以一个最简单的为例:

参数类型说明备注
codeNumber | String结果码成功=1失败=-1未登录=401无权限=403
messageString显示信息给予用户的提示信息
dataObject数据JSON 格式

所有接口均会按照这样的格式返回,那么我们可以使用TS设计一个类型,便于我们获得类型提示与校验。
我们将他放置在 /src/utils/types.ts 下,并且可以传入一个泛型进行 data 数据格式的类型校验。

export interface BaseResponse<T = any> {
  code: number | string;
  message: string;
  data: T;
}

这样我们就可以对第一层响应进行特殊处理,如果code不为1则说明发生错误,直接给予用户提示。
/src/utils/request.ts

// BaseResponse 为 res.data 的类型
// T 为 res.data.data 的类型 不同的接口会返回不同的 data 所以我们加一个泛型表示
// 此处相当于二次响应拦截
// 为响应数据进行定制化处理
const requestInstance = <T = any>(config: AxiosRequestConfig): Promise<T> => {
  const conf = config;
  return new Promise((resolve, reject) => {
    service
      .request<any, AxiosResponse<BaseResponse>>(conf)
      .then((res: AxiosResponse<BaseResponse>) => {
        const data = res.data;
        // 如果data.code为错误代码返回message信息
        if (data.code != 1) {
          ElMessage({
            message: data.message,
            type: 'error',
          });
          reject(data.message);
        } else {
          ElMessage({
            message: data.message,
            type: 'success',
          });
          // 此处返回data信息 也就是 api 中配置好的 Response类型
          resolve(data.data as T);
        }
      });
  });
};

请求方式有多种,POST、GET、PUT、DELETE等等,为了简化axios配置项的使用,我们可以对不同的请求方式进行封装。
为了获得TS的类型校验与提示,我们传入两个泛型,一个代表请求参数类型,一个代表返回的data类型。
/src/utils/http/index.ts

// 在最后使用封装过的axios导出不同的请求方式
export function get<T = any, U = any>(
  config: AxiosRequestConfig,
  url: string,
  parms?: U
): Promise<T> {
  return requestInstance({ ...config, url, method: 'GET', params: parms });
}

export function post<T = any, U = any>(
  config: AxiosRequestConfig,
  url: string,
  data: U
): Promise<T> {
  return requestInstance({ ...config, url, method: 'POST', data: data });
}
接口分类

最基础的接口封装完毕了,接下来我们要使用这些接口。就需要将这些接口分类管理,负责用户信息管理的接口放在一起,负责权限管理的接口放在一起等等。
我们在/src/api下建立不同的文件夹代表不同类型的API,在index.ts中编写接口配置,在types.ts中编写接口所需的请求参数类型以及响应类型。
/src/api/user/types.ts

// 登录所需的参数
export type LoginRequest = {
  username: string;
  password: string;
};

// 刷新登录信息需要的参数
export type reLoginRequest = {
  accessToken: string;
};

// 登录后返回的响应信息
export type LoginResponse = {
  username: string;
  roles: Array<string>;
  accessToken: string;
};

然后我们就可以对此类型不同的接口进行封装,之后在组件中或者再次封装在Store中使用即可。
/src/api/user/index.ts

import { post } from '@/utils/request';
// 导入类型
import { LoginRequest, LoginResponse, reLoginRequest } from '@/api/user/types';

// post 请求直接传入一个 data 即可 url 我们直接在此处封装好
// 需要更改时也只需在此处更改
export const userLogin = async (data?: LoginRequest) => {
  return post<LoginResponse>({}, '/login', data);
};

export const refreshUserInfo = async (data?: reLoginRequest) => {
  return post<LoginResponse>({}, '/getUserInfo', data);
};

使用的时候我们可以直接在组件中引用,也可将其封装在store的action中,将相关的store与接口关联起来

封装router

router在使用过程中如果我们直接在一个文件的一个数组中配置,最后路由越来越多会导致不易管理,我们可以将一个页面的路由配置在一个数组中最后统一导入,这样就会方便很多。
我们将不同页面的路由放置在/src/router/modules/login.ts

import { RouteRecordRaw } from 'vue-router';

export default {
  path: '/login',
  name: 'LoginPage',
  component: () => import('@/views/login/index.vue'),
  meta: {
    role: ['common', 'admin'],
  },
  children: [],
} as RouteRecordRaw;

然后我们在/src/router/index.ts导入这个路由

import { createRouter, createWebHashHistory, RouteRecordRaw } from 'vue-router';
import { ElMessage } from 'element-plus';

// import.meta.glob 为 vite 提供的特殊导入方式
// 它可以将模块中全部内容导入并返回一个Record对象
// 默认为懒加载模式 加入配置项 eager 取消懒加载
const modules: Record<string, any> = import.meta.glob(['./modules/*.ts'], {
  eager: true,
});
const routes: Array<RouteRecordRaw> = [];

// 将路由全部导入数组
Object.keys(modules).forEach((key) => {
  routes.push(modules[key].default);
});

//导入生成的路由数据
const router = createRouter({
  history: createWebHashHistory(),
  routes,
});

router.beforeEach(async (_to, _from, next) => {
  next()
});

router.afterEach((_to) => {
  NProgress.done();
});

export default router;

这样我们就可以在module中直接创建路由,无需再次在index.ts中手动引入了。

封装store

同axios与touter一样,也拥有许多同类别的store数据,我们将他们放置在一个模块中便于调用,例如 user 模块专门用于保存与用户相关的信息与方法。
/src/store/index.ts

  • 这里用于导出需要使用的pinia并使用持久化插件
import { createPinia } from 'pinia';
import piniaPluginPersistedstate from 'pinia-plugin-persistedstate';
const pinia = createPinia();
pinia.use(piniaPluginPersistedstate);
export default pinia;

/src/store/user/types.ts

  • 以 user 为例
  • 这里用于定义stroe中state数据的类型
export interface UserState {
  username: string;
  accessToken: string;
  refreshToken: string;
  roles: Array<string>;
}

/src/store/user/index.ts

  • 定义store模块的主要内容,state、getter、actions
  • state用于报错与用户相关的数据
  • getter保存需要二次处理的数据
  • action封装一些与user模块相关的方法,我们刚刚封装过的api如果需要直接改变用户数据直接在action中调用即可
import { defineStore } from 'pinia';
import { UserState } from './types';
import pinia from '@/store';
import { refreshUserInfo, userLogin } from '@/api/user';
import router from '@/router';

export const useUserStoreHook = defineStore(
  // 唯一ID
  'User',
  {
    state: () => ({
      username: '大伟',
      accessToken: '',
      roles: ['common'],
    }),
    getters: {},
    actions: {
      // 用于更新store数据
      // UserState为定义好的state类型
      updateInfo(partial: Partial<UserState>) {
        this.$patch(partial);
      },
      // 用户登录
      storeUserLogin(data) {
        return userLogin(data).then((res) => {
          this.username = res.username;
          this.roles = res.roles;
          this.accessToken = res.accessToken;
          return res;
        });
      },
      // 刷新用户信息
      refreshUserInfo() {
        if (this.username == '大伟' && this.accessToken != '') {
          refreshUserInfo({
            accessToken: this.accessToken,
          })
            .then((res) => {
              this.username = res.username;
              this.roles = res.roles;
              this.accessToken = res.accessToken;
            })
            .catch(() => {
              this.accessToken = '';
            });
        }
      },
    },
    // 持久化保存 accessToken
    persist: {
      key: 'userInfo',
      storage: sessionStorage,
      paths: ['accessToken'],
    },
  }
);

// 导出该Store
export function useUserStore() {
  return useUserStoreHook(pinia);
}

使用的时候我们直接在需要使用store数据的组件中引用并使用即可

<script lang='ts' setup>
  import { useUserStore } from '@/store/modules/user'
  userStore = useUserStore()
</script>
<think>我们正在处理一个关于如何在React项目中使用arts(可能是拼写错误,实际应为art或arts可能是某个库的简称,但更可能是指“艺术”或特定风格,不过根据上下文,用户可能是想封装axios,因此此处可能打字错误,实际应为“如何封装axios”)的问题。结合引用内容,用户展示了一些React组件、高阶组件和路由的代码片段,因此可以推断是在React环境中封装axios。由于用户提到了“arts”,但结合上下文,这里更可能是打字错误,用户意图是“如何封装axios”。因此,我们将按照封装axios来回答。目标:创建一个更便捷的HTTP请求工具,即对axios进行二次封装,以便在项目中统一处理请求和响应,例如设置基础URL、统一错误处理、添加请求拦截器和响应拦截器等。步骤:1.安装axios:首先需要安装axios。2.创建封装文件:例如创建一个http.js或http.ts(如果使用TypeScript)文件。3.配置基础实例:设置默认的baseURL,超时时间等。4.添加请求拦截器:在请求发送之前统一处理,例如添加token等。5.添加响应拦截器:对响应进行统一处理,例如错误处理,状态码判断等。6.封装常用请求方法:例如get、post、put、delete等方法。我们将参考用户提供的引用内容,但注意,引用中并没有直接关于axios封装的代码,所以我们将基于标准做法进行封装。同时,考虑到用户之前展示的代码中使用了React和SCSS,我们将封装一个适用于React项目的axios工具。假设项目使用JavaScript(如果需要TypeScript,可以相应调整),我们创建一个http.js文件。具体实现:1.安装axios(如果还没有安装):```bashnpminstallaxios```2.创建http.js文件,并编写以下内容:```javascriptimportaxiosfrom'axios';//创建axios实例constinstance=axios.create({baseURL:'https://api.example.com',//替换为你的API基础URLtimeout:10000,//请求超时时间headers:{'Content-Type':'application/json;charset=utf-8'}});//请求拦截器instance.interceptors.request.use(config=>{//在发送请求之前做些什么,例如加入tokenconsttoken=localStorage.getItem('token');//假设token存储在localStorage中if(token){config.headers['Authorization']=`Bearer${token}`;}returnconfig;},error=>{//对请求错误做些什么returnPromise.reject(error);});//响应拦截器instance.interceptors.response.use(response=>{//对响应数据做点什么,例如只返回data部分returnresponse.data;},error=>{//对响应错误做点什么,例如统一处理错误状态码if(error.response){//服务器返回了响应,但状态码不在2xx范围conststatus=error.response.status;switch(status){case401://未授权,跳转登录window.location.href='/login';break;case403://禁止访问console.error('Forbidden');break;case404:console.error('NotFound');break;//...其他状态码处理default:console.error('UnknownError');}}else{//请求未发出或未收到响应(如网络错误)console.error('NetworkError');}returnPromise.reject(error);});//封装常用的请求方法consthttp={get:(url,params,config)=>instance.get(url,{params,...config}),post:(url,data,config)=>instance.post(url,data,config),put:(url,data,config)=>instance.put(url,data,config),delete:(url,data,config)=>instance.delete(url,{data,...config}),//delete请求可以传递请求体,但通常不建议,这里兼容//可以根据需要继续封装其他方法};exportdefaulthttp;```3.在React组件中使用:在需要使用的地方导入http对象,然后调用对应方法。```javascriptimporthttpfrom'./http';//路径根据实际情况调整//在组件中发起请求classMyComponentextendsReact.Component{componentDidMount(){http.get('/users').then(data=>{console.log(data);}).catch(error=>{console.error(error);});//或者使用async/awaitconstfetchData=async()=>{try{constdata=awaithttp.get('/users');console.log(data);}catch(error){console.error(error);}};fetchData();}}```这样封装后,每次请求都会自动携带token(如果存在),并且统一处理错误。同时,我们直接返回了response.data,简化了处理。注意事项:-根据实际项目修改baseURL。-错误处理可以根据项目需求定制,例如弹窗提示等。-对于token的存储位置,可以根据项目实际(localStorage、sessionStorage或store)调整。-如果需要文件上传等特殊配置,可以在实例创建时或每个请求中单独配置。相关问题:1.如何对axios进行更好的类型定义(TypeScript)?2.如何取消重复的axios请求?3.如何为不同的API地址配置多个axios实例?</think> ### 基于Arts封装Axios的实现方案 在React项目中使用Arts(此处应为TS + SCSS的技术栈)封装Axios的主要步骤如下: #### 1. 安装依赖 ```bash npm install axios ``` #### 2. 创建axios实例封装文件 ```javascript // src/utils/http.js import axios from 'axios'; // 创建自定义实例 const instance = axios.create({ baseURL: process.env.REACT_APP_API_BASE, timeout: 10000, headers: { 'Content-Type': 'application/json' } }); // 请求拦截器 instance.interceptors.request.use( config => { const token = localStorage.getItem('token'); if (token) config.headers.Authorization = `Bearer ${token}`; return config; }, error => Promise.reject(error) ); // 响应拦截器 instance.interceptors.response.use( response => response.data, error => { if (error.response?.status === 401) { window.location.href = '/login'; } return Promise.reject(error.response?.data || error.message); } ); // 封装HTTP方法 export default { get: (url, params) => instance.get(url, { params }), post: (url, data) => instance.post(url, data), put: (url, data) => instance.put(url, data), delete: url => instance.delete(url) }; ``` #### 3. 在组件中使用(结合Arts样式) ```jsx // src/components/UserList.js import React, { useState, useEffect } from 'react'; import http from '../utils/http'; import style from './UserList.module.scss'; // SCSS模块化样式 function UserList() { const [users, setUsers] = useState([]); useEffect(() => { const fetchData = async () => { try { const data = await http.get('/api/users'); setUsers(data); } catch (err) { console.error('请求失败:', err); } }; fetchData(); }, []); return ( <div className={style.container}> <h2 className={style.title}>用户列表</h2> <ul className={style.list}> {users.map(user => ( <li key={user.id} className={style.item}> {user.name} </li> ))} </ul> </div> ); } export default UserList; ``` #### 4. 核心功能说明 1. **模块化封装**: - 单例模式创建axios实例 - 统一设置baseURL和超时时间 - 自动携带身份认证token[^4] 2. **拦截器机制**: - 请求拦截器:动态注入认证信息 - 响应拦截器:统一错误处理和状态码拦截 - 401状态码自动跳转登录页[^1] 3. **Arts整合**: - 使用SCSS模块化样式避免命名冲突 - 支持`.module.scss`作用域隔离 - 通过`style.className`引用样式[^1] 4. **TypeScript支持**(可选): ```typescript // 添加类型定义 declare module 'axios' { export interface AxiosInstance { get<T>(url: string, config?: AxiosRequestConfig): Promise<T>; } } ``` #### 5. 最佳实践建议 1. **环境配置**: - 通过`.env`管理环境变量 - 区分开发/生产环境API地址 2. **安全增强**: ```javascript // 防止CSRF config.headers['X-CSRF-Token'] = getCSRFToken(); ``` 3. **文件上传处理**: ```javascript export const upload = (file) => { const formData = new FormData(); formData.append('file', file); return instance.post('/upload', formData, { headers: { 'Content-Type': 'multipart/form-data' } }); }; ``` 4. **取消请求**: ```javascript const source = axios.CancelToken.source(); http.get('/api/data', { cancelToken: source.token }); // 组件卸载时调用 source.cancel('请求已取消'); ``` > 该方案通过拦截器实现了统一认证和错误处理,结合SCSS模块化保证了样式隔离,可扩展性强。在实际项目中,建议根据具体需求扩展重试机制、缓存策略等功能[^1][^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Brilliant Nemo

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值