封装axios

这段代码展示了如何使用axios创建一个HTTP服务实例,并配置了请求和响应拦截器来处理token验证、超时、错误提示等问题。同时,提供了方便的方法如asyncGet、asyncPost等用于发起HTTP请求,以及文件下载和读取本地文件的功能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

import axios from 'axios';
import qs from 'qs';
import vMessage from '@/components/messageTips';

const service = axios.create({
  // 请求超时时间
  // timeout: 3000
});
// service.defaults.withCredentials = true;

// 请求拦截器
service.interceptors.request.use(
  async config => {
    if (config.url.indexOf('login') > 0) { // 部分接口不需要验证token,如登录
      // 下面5个数据  都是登陆后就放入localStorage的
      let tenantToken = localStorage.getItem('tenant_token'); // token
      const tenantId = localStorage.getItem('tenantId'); // 业务所需
      const userId = localStorage.getItem('userId'); // 业务所需
      const expTime = localStorage.getItem('exp_time'); // token失效时间点(ms)
      const refreshTime = localStorage.getItem('refresh_time'); // 时间长度(ms)
      if (expTime !== 'permanent') {
        // 校验token有效期
        let nowTime = '';
        const [err, res] = await to(axios.head(`${window.projectConfig.asiaInfoUrl}/tenant/time`));
        if (!err && res) {
          nowTime = new Date(res.headers.date).getTime();
          // 若token有效期到今日10点,refreshTime=5分钟,若用户在9:55-10:00之间仍然在调用接口,则重新获取新的token,并保存新token的信息,下次请求中携带新token,可避免用户操作过程中,突然返回登录页的尴尬
          // 当前时间>失效时间-token刷新时间,获取新token,失效时间重新开始计算,此过程用户无感知
          if (nowTime > expTime - refreshTime && nowTime < expTime) {
            const [err2, res2] = await asyncPost('tenant/t_refresh_token', qs.stringify({
              tenant_id: tenantId,
              user_id: userId,
              token: tenantToken
            }));
            if (!err2 && res2) {
              tenantToken = res2.data.result.tenant_refreshToken;
              localStorage.setItem('tenant_token', tenantToken);
              localStorage.setItem('exp_time', res2.data.result.exp_time);
            }
          }
        }
      } else {
        // 当expTime==='permanent' 不需要校验token有效期
      }
      config.headers.tenant_token = tenantToken;
    }
    return config;
  },
  err => {
    console.log(err);
  }
);

service.interceptors.response.use(
  response => {
    if (response.status === 200) {
      return response;
    } else {
      // router.replace('/403');
    }
  },
  err => {
    // console.log(err.request);
    if (err.request.status === 510) {
      vMessage.error('token校验异常,请重新登录');
      localStorage.clear();
      // 退出登录时,需要将vuex中的变量全部还原为初始化值,如果用$router跳转无法还原
      window.location = 'login';
      // return Promise.resolve(err.response);
      // return err.response;
    } else {
      return Promise.reject(err);
    }
  }
);

const formatUrl = url => {
  let returnUrl = '';
  if (url.startsWith('http://') || url.startsWith('https://')) {
    returnUrl = url;
  } else {
    // 传递的url不能以 '/' 开头,否则拼接后的url会存在双斜杠,可能会导致服务器报错
    if (url.startsWith('/')) {
      console.error(`url=${url},此url错误,请勿以'/'开头,`);
      returnUrl = 'error';
    } else {
      returnUrl = `${window.projectConfig.asiaInfoUrl}/${url}`;
    }
  }
  return returnUrl;
};

// 此方法是为了方便使用axios.all
const getPromise = (url, params, config) => {
  const newUrl = formatUrl(url);
  if (newUrl === 'error') {
    return;
  }
  const obj = Object.assign({
    url: newUrl,
    method: 'get',
    params
  }, config);
  return service(obj);
};

const asyncGet = (url, params, config, errorExt) => {
  const promise = getPromise(url, params, config);
  return to(promise, errorExt);
};

// 此方法是为了方便使用axios.all
const downloadPromise = (url, params, config) => {
  const newUrl = formatUrl(url);
  if (newUrl === 'error') {
    return;
  }
  const obj = Object.assign({
    url: newUrl,
    method: 'get',
    params,
    responseType: 'arraybuffer'
  }, config);
  return service(obj);
};

const asyncDownload = (url, params, config, errorExt) => {
  const promise = downloadPromise(url, params, config);
  return to(promise, errorExt);
};

// 此方法是为了方便使用axios.all
const postPromise = (url, data, config) => {
  const newUrl = formatUrl(url);
  if (newUrl === 'error') {
    return;
  }
  const obj = Object.assign({
    url: newUrl,
    method: 'post',
    data
  }, config);
  return service(obj);
};

const asyncPost = (url, data, config, errorExt) => {
  const promise = postPromise(url, data, config);
  return to(promise, errorExt);
};

// 此方法是为了方便使用axios.all
const putPromise = (url, data, config) => {
  const newUrl = formatUrl(url);
  if (newUrl === 'error') {
    return;
  }
  const obj = Object.assign({
    url: newUrl,
    method: 'put',
    data
  }, config);
  return service(obj);
};

const asyncPut = (url, data, config, errorExt) => {
  const promise = putPromise(url, data, config);
  return to(promise, errorExt);
};

// 此方法是为了方便使用axios.all
const deletePromise = (url, params, config) => {
  const newUrl = formatUrl(url);
  if (newUrl === 'error') {
    return;
  }
  const obj = Object.assign({
    url: newUrl,
    method: 'delete',
    params
  }, config);
  return service(obj);
};

const asyncDelete = (url, params, config, errorExt) => {
  const promise = deletePromise(url, params, config);
  return to(promise, errorExt);
};

// 读取本地文件
const readLocalFile = (url, params, config, errorExt) => {
  const obj = Object.assign({
    url: url,
    method: 'get',
    params
  }, config);
  return to(service(obj), errorExt);
};

const to = (promise, errorExt = '未知错误') => {
  return promise
    .then(function (data) {
      return [null, data];
    })
    .catch(function (err) {
      if (errorExt) {
        Object.assign(err, { errorExt });
      }
      if (err.code === 'ECONNABORTED' && err.message.indexOf('timeout') !== -1) {
        vMessage.error(`请求超时 [ 错误编码:${err.request.status} ]`);
      } else if (err.message.indexOf('Network Error') !== -1) {
        vMessage.error(`网络异常 [ 错误编码:${err.request.status} ]`);
      } else if (err.request.status === 403) {
        vMessage.error(`无访问权限 [ 错误编码:${err.request.status} ]`);
      } else if (err.request.status === 404) {
        vMessage.error(`请求资源不存在 [ 错误编码:${err.request.status} ]`);
      } else if (err.request.status === 460) {
        vMessage.error(`token错误 [ 错误编码:${err.request.status} ]`);
      } else {
        if (err.request.status !== undefined) {
          vMessage.error(`系统繁忙,请稍后再试 [ 错误编码:${err.request.status} ]`);
        } else {
          vMessage.error('系统繁忙,请稍后再试');
        }
      }
      return [err, undefined];
    });
};

const handleStream = function (res, fileName) {
  const blob = new Blob([res], { type: 'application/octet-stream' });
  if (typeof window.navigator.msSaveBlob !== 'undefined') {
    window.navigator.msSaveBlob(blob, fileName);
  } else {
    const URL = window.URL || window.webkitURL;
    const objectUrl = URL.createObjectURL(blob);
    const a = document.createElement('a');
    // safari doesn't support this yet
    if (typeof a.download === 'undefined') {
      window.location = objectUrl;
    } else {
      a.href = objectUrl;
      a.download = fileName;
      document.body.appendChild(a);
      a.click();
      a.remove();
    }
  }
};

const exportObj = {
  service,
  to,
  getPromise,
  postPromise,
  putPromise,
  deletePromise,
  asyncGet,
  asyncPost,
  asyncPut,
  asyncDelete,
  asyncDownload,
  handleStream,
  readLocalFile
};

export default exportObj;

<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]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值