Axios封装示例代码+封装思路

Author: Echo
Time: 2019-09-20 10:50:39

axios文档地址

在vue项目中,和后台交互获取数据这块,我们通常使用的是axios库,它是基于promise的http库,可运行在浏览器端和node.js中。他有很多优秀的特性,例如拦截请求和响应、取消请求、转换json、客户端防御cSRF等。所以我们的尤大大也是果断放弃了对其官方库vue-resource的维护,直接推荐我们使用axios库.

本次封装以vue项目为例做实践。

安装
$ npm install axios
引入

在项目的src目录下新建文件夹request,文件夹下新建http.js和api.js文件:

  • http.js: 封装axios
  • api.js: 用来统一管理接口
// 在http.js中引入axios
import axios from 'axios'; // 引入axios
import QS from 'qs'; // 引入qs模块,用来序列化post类型的数据,后面会提到

环境的切换->配置不同的baseURL

我们的项目环境可能有开发环境、测试环境和生产环境.
我们通过node的环境变量来匹配我们的默认的接口url前缀
axios.defaults.baseURL可以设置axios的默认请求地址

const baseURL_Map = {
  dev:'https://www.baidu.com',
  uat: 'https://www.ceshi.com',
  product: 'https://www.production.com'
}
axios.defaults.baseURL = baseURL_Map[process.env.NODE_ENV] 

扩展:

process.env.NODE_ENV
  • process对象是全局变量: 提供当前node.js的有关信息,以及控制当前node.js的有关进程。因为是全局变量,它对于node应用程序是始终可用的,无需require()
  • process.env属性返回一个包含用户环境信息的对象
  • process.env.NODE_ENV:不是process.env对象上原有的属性,是我们自己添加进去的一个环境变量,用来确定当前所处的开发阶段。
  • 测试环境: uat; 生产环境:product; 开发环境: dev
设置请求超时变量
// 例如超过了10s,就会告知用户当前请求超时,请刷新等。

const TIMEOUT = 30000;

扩展:

  • 统一处理所有接口的超时时间: axios.defaults.timeout = 10000;
设置post请求头
axios.defaults.headers.post['Content-Type'] = 'application/x-www-form-urlencoded;charset=UTF-8';

此处无需设置,在下面封装的时候会针对get和post请求做统一对应处理

定义HTTP响应状态码的范围
/**
 * 判断 HTTP 请求是否成功
 * @param {number} status - HTTP 状态码
 * @returns {Boolean}
 */
const isBadRequest = status => status >= 300;

映射 HTTP 错误时的提示语句

/**
 * 映射 HTTP 错误时的提示语句
 * @param {enhanceError} error
 * @returns {Object}
 */
const badStatusMap = error => ({
  400: '请求错误',
  401: '未授权,请登录',
  403: '拒绝访问',
  404: `请求地址出错 ${error.response ? error.response.config.url : ''}`,
  408: '请求超时',
  500: '服务器内部错误',
  501: '服务未实现',
  502: '网关错误',
  503: '服务不可用',
  504: '网关超时',
  505: 'HTTP版本不受支持'
});

封装axios
  • http.js
/**
 * http.js
 * @param {Object} [globalOption]
 */
import {
  mergeWith, omit, isPlainObject
} from 'lodash';

// isPlainObject : checks if value is a plain object, that is, an object created by the Object constructor or one with a [[Prototype]] of null
// mergeWith 是用于递归合并来源对象的自身和继承的可枚举属性到目标对象。 
// 基本意思就是将默认的option配置和手动配置的globalOption,通过递归的形式合并成一个option.


/**
 * 修复 axios 转换 bug,如可能把请求中的日期等转成空对象的问题
 *  'utils.merge' in 'node_modules\axios\lib\core\Axios.js'
 * @param {any} obj
 */
const fixAxiosDateMerge = (obj) => {
  if (!obj) return obj;
  if (obj instanceof Date) return obj.getTime();
  if (!isPlainObject(obj)) return obj;
  Object.keys(obj).forEach((key) => {
    obj[key] = fixAxiosDateMerge(obj[key]);
  });
  return obj;
};

function axiosWrapper(globalOption) {
  return function(option) {
    // 此处的操作相对与开放一点,把baseURL、timeout、responseType、可以根据接口不同做单独配置。
    option = mergeWith({
      baseURL: BASE_URL,
      headers: {
        'Content-Type': 'application/x-www-form-urlencoded',
        'Cache-Control': 'no-cache'
      },
      timeout: TIMEOUT,
      withCredentials: true,
      responseType: '',
      validateStatus: status => !isBadRequest(status)

      // 可以定义HTTP响应状态码的范围。如果`validateStatus`返回`true`(或被设置为`null` promise将被解析;否则,promise将被拒绝。
    }, globalOption, option);

  
    if (option.data) {
      option.data = fixAxiosDateMerge(option.data);
    }
    if (option.params) {
      option.params = fixAxiosDateMerge(option.params);
    }

    const r = axios.create(option);
    
    // 在axios的源代码中,超时后, 会在拦截器那里 axios.interceptors.response 捕抓到错误信息, 且 error.code = "ECONNABORTED"

    // timeout
    r.interceptors.request.use(
      config => config,
      (error) => {
        
        if (error && error.code === 'ECONNABORTED'
          && ~error.message.indexOf('timeout')) {
          console.log('[AxiosWrapper] request timeout');
        }
        return Promise.reject(error);
      }
    );

    // other errors // 跳转到对应的报错页面
    r.interceptors.request.use(
      config => config,
      (error) => {
        const errorInfo = error.response;
        if (errorInfo) {
          const errorStatus = errorInfo.status;
          router.push({
            path: `/error/${errorStatus}`
          });
        }
        return Promise.reject(error);
      }
    );

    // make data
    r.interceptors.request.use(
      (opt) => {
        const params = mergeWith({}, opt.data, opt.params); // cloneDeep(opt.data);
        opt = omit(opt, ['data', 'params']);
        const needBody = /^(put|post|patch)$/i.test(opt.method);
        const sendJSON = opt.headers
          && opt.headers['Content-Type'] === 'application/json';
          // 根据请求头和method来处理入参的格式
        if (needBody) {
          opt.data = sendJSON
            ? JSON.stringify(params)
            : qs.stringify(params);
        } else {
          opt.params = params;
        }
        return opt;
      }
    );

    // bad HTTP request
    r.interceptors.response.use(
      response => response,
      (error) => {
        if (error && error.response) {
          const { status, statusText } = error.response;
          if (isBadRequest(status)) {
            console.warn('[AxiosWrapper] bad HTTP request: status is %s \n', status, error.response);
            error.message = badStatusMap(error)[status] || statusText;
            if (!error.message) {
              try {
                error.message = JSON.parse(error.response.request.responseText).errmsg;
              } catch (ex) {
                console.log('[AxiosWrapper] json parse error', ex);
              }
            }
            Vue.prototype.Config.ROOT_VUE.$message.error(error.message);
            return Promise.reject(new Error(error.message));
          }
        }
        return Promise.reject(error);
      }
    );

    // check business logic
    // 请求失败,根据HTTP响应状态码做出对应的messageTip
    r.interceptors.response.use(
      (response) => {
        if (option.responseType === 'arraybuffer') {
          return response;
        }
        const [reqURL, resData] = takeapartResponse(response);
        const noWarn = 'no-global-config-warn' in response.config.headers
          && !!response.config.headers['no-global-config-warn'];
        const isValid = isValidCode(resData.errno);
        const method = response.config.method.toUpperCase();
        if (!isValid) {
          // eslint-disable-next-line prefer-promise-reject-errors
          return Promise.reject({
            type: ERR_CODE,
            reqURL,
            resData,
            noWarn,
            method
          });
        }
        const reqStr = `${method} ${reqURL}`;
        // warnByResponse 函数是根据业务针对不同接口返回不同的错误提示。这个函数可以根据业务自行处理。
        warnByResponse(reqStr, resData, true, noWarn);
        return resData;
      }
    );
    

    // wrong business logic
    // r.interceptors.response.use(
    //   response => response,
    //   (error) => {
    //     if (error.type === ERR_CODE) {
    //       const reqStr = `${error.method} ${error.reqURL}`;
    //       warnByResponse(reqStr, error.resData, false, error.noWarn);
    //     }
    //     return Promise.reject(error.resData);
    //   }
    // );

    return r(option);
  };
}

简单的封装axios函数

/**
 * 获得一般的 wrapper
 * @param {Object} [option]
 */
 // `transformResponse` 在传递给 then/catch 前,允许修改响应数据
export default option => axiosWrapper({
  transformResponse: data => void (0), // eslint-disable-line no-unused-vars
  
})(option);

在使用axios时,注意到配置选项中包含params和data两者,以为他们是相同的,实则不然。

因为params是添加到url的请求字符串中的,用于get请求。

而data是添加到请求体(body)中的, 用于post请求。

api.js使用实例:
import axios, { download } from './http.js';
export const mock = params => axios({
    url: '/mock',
    method: 'get',
    params
});

export const upload = data => axios({
    url: '/upload',
    method: 'post',
    data
});
用于下载的axiosWrapper
export const download = option => axiosWrapper({
  responseType: 'arraybuffer'
})(option)
  .then((res) => {
    const contentType = res.headers['content-type'];

    if (~contentType.indexOf('json')) {
      const resData = res.data || res.request.response;
      const text = Buffer.from(resData).toString('utf8');
      const json = JSON.parse(text);
      return Promise.reject(json);
    }

    const disposition = res.headers['content-disposition'];
    if (disposition && disposition.match(/attachment/)) {
      let filename = disposition.replace(/attachment;.*filename=/, '').replace(/"/g, '');
      filename = filename && filename !== ''
        ? filename
        : (option.defaultFilename || 'noname');
      const blob = new Blob([res.data], { type: contentType });
      saveAs(blob, filename);
    }

    return Promise.resolve(res);
  });

api.js使用实例:

import axios, { download } from './http';

export const export = (data, defaultFilename) = download({
  url: '/api/export',
  method: 'post',
  data,
  defaultFilename
})

封装一个axios的思路

  • 配置请求基本信息的常量:(用于基本处理)
    BASE_URL :根据环境变量,针对生成、测试、开发环境配置对应的url
    TIMEOUT: 配置接口请求超时时间。
    headers: 配置请求头
    validateStatus: 定义HTTP响应状态码的范围。如果validateStatus返回true(或被设置为null promise将被解析;否则,promise将被拒绝。

可以请求接口的不同,封装axios的时候,将 请求基本信息的常量 通过 参数 的方式单独进行配置。

扩展:
具体方法:通过lodash的mergeWith方法将默认的option配置和手动配置的globalOption,通过递归的形式合并成一个option。

  • 请求拦截

timeout: 在axios的源代码中,超时后, 会在拦截器那里 axios.interceptors.response 捕抓到错误信息, 且 error.code = "ECONNABORTED",可以根据这个code提示超时报错信息。

token:统一在http请求的header都加上token,这样后台根据token判断你的登录情况,即使本地存在token,也有可能token是过期的,所以在响应拦截器中要对返回状态进行判断

other errors: 根据请求状态status,直接promise.reject.路由跳转到对应的报错页面。
根据请求头和method来处理入参的格式

在使用axios时,注意到配置选项中包含params和data两者,以为他们是相同的,实则不然。
因为params是添加到url的请求字符串中的,用于get请求。
而data是添加到请求体(body)中的, 用于post请求。

  • 响应拦截
    badStatusMap: 请求失败,根据HTTP响应状态码做出对应的messageTip
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,616评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,020评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,078评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,040评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,154评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,265评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,298评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,072评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,491评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,795评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,970评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,654评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,272评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,985评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,815评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,852评论 2 351