Axios 源码浅析(一)—— Axios 实例及配置

863 阅读7分钟

引言

熟悉 Vue 的同学们,肯定都用过 Axios , 简洁的 API 语法和方便的拦截器都受到了开发者的热烈欢迎,使用背后,是否也想了解一下实现原理呢,其实源码并不难阅读,这里我就抛砖引玉,对源码做一个浅析

本文面向经常使用 Axios 的开发者,故以下内容默认阅读者都熟悉或精通 Axios,不会对 Axios 功能上做详细说明

本文为系列的第一部分,将对 Axios 的 export 实例以及默认配置两个部分进行分析

本文分析的 Axios 版本为 0.21.0

回顾

首先我们回顾一下常用的 Axios 功能都有哪些

  • axios(config) / axios.get / axios.post / axios.delete ....

    调用 axios ,传参发起一个请求,或者直接发起 get、post、delete 等请求

  • axios.create

    创建一个新的 axios 实例

  • axios.defaults.xxxx

    配置 axios 的默认配置

  • axios.interceptors.request / axios.interceptors.response

    配置 axios 的全局拦截器,也可以只配置某个通过 create 创建的实例的拦截器

  • cancel

    取消某个请求

接下来我们根据源码一一解析,上述的功能是如何实现的

代码结构

源代码里的主要目录是 lib ,其余的是一些单元测试、文档之类的,这里不再赘述

lib
│  axios.js
│  defaults.js
│  utils.js
│
├─adapters
│      http.js
│      README.md
│      xhr.js
│
├─cancel
│      Cancel.js
│      CancelToken.js
│      isCancel.js
│
├─core
│      Axios.js
│      buildFullPath.js
│      createError.js
│      dispatchRequest.js
│      enhanceError.js
│      InterceptorManager.js
│      mergeConfig.js
│      README.md
│      settle.js
│      transformData.js
│
└─helpers
        bind.js
        buildURL.js
        combineURLs.js
        cookies.js
        deprecatedMethod.js
        isAbsoluteURL.js
        isAxiosError.js
        isURLSameOrigin.js
        normalizeHeaderName.js
        parseHeaders.js
        README.md
        spread.js

这里有几个重要文件(axios.js、defaults.js、utils.js)以及几类文件分别放在四个文件夹里

  • axios.js

    该文件作为入口文件,主要作为生成并导出 axios 对象、以及扩展一系列例如 axios.create 的功能方法

  • defaults.js

    该文件是默认配置

  • utils.js

    该文件是一系列辅助方法,例如:isStirng、extend 等等

  • adapter

    该目录里主要存放一些与 ajax 的适配器,也就是封装原生的 xmlHttpRequest 或者 node 的 http 库等,然后处理成方便使用的自己的方法,如果以后有更新的 ajax 方法,例如 fetch,那么 Axios 只需要修改这部分去适配新的接口即可

  • cancel

    这里实现取消请求的功能

  • core

    这是 Axios 的核心代码,包括 Axios 的基类、一些错误的封装等等,其中比较核心的又有 Axios.js、dispatchRequest.js、InterceptorManager.js

  • helpers

    这里另一部分辅助的功能性模块,和 utils.js 的区别可能是因为 utils.js 是更加通用的方法,而这个目录里的则是定制的一些辅助方法

adapter 目录的作用,是因为使用了设计模式里的适配器模式,有兴趣的童鞋可以延伸阅读一下

源码分析

axios(config) / axios.get / axios.create 等方法实现

我们暂时先不关心具体的实现,首先来看如何将抽象方法暴露出来提供开发者使用,这里主要关注 lib/axios.js 这个文件

// path: lib/axios.js

var utils = require('./utils');
var bind = require('./helpers/bind');
var Axios = require('./core/Axios');
var mergeConfig = require('./core/mergeConfig');
var defaults = require('./defaults');

function createInstance(defaultConfig) {
  var context = new Axios(defaultConfig);
  var instance = bind(Axios.prototype.request, context);

  // Copy axios.prototype to instance
  utils.extend(instance, Axios.prototype, context);

  // Copy context to instance
  utils.extend(instance, context);

  return instance;
}

// Create the default instance to be exported
var axios = createInstance(defaults);

// Factory for creating new instances
axios.create = function create(instanceConfig) {
  return createInstance(mergeConfig(axios.defaults, instanceConfig));
};

axios.Cancel = ...
axios.CancelToken = ...
axios.isCancel = ...
axios.all = ...
axios.spread = ...
axios.isAxiosError = ...

module.exports = axios;

可以看出,我们开发者使用的 axios,也是通过 Axios 基类生成的一个实例(严格意义上说并不是一个单纯的实例)

可能有的同学要问了,为何不直接 export 一个实例呢,还封装一个函数,有什么用意呢,其实这里的目的是在导出实例的同时,将我们标题上说的 axios(config) / axios.get / axios.create 等方法也绑定上去,扩展这个实例,使它的功能更强大,我们接下来一步步看是怎么实现的:

  • var context = new Axios(defaultConfig);
    var instance = bind(Axios.prototype.request, context);
    

    这两行先是生成一个实例 context,然后通过 bind 函数将基类 Axiosrequest 方法的上下文 this 绑定为实例 context,并生成一个新的方法 instance,也就是未来我们用的 axios 对象,到这一步,axios 对象本质上还只是一个 request 方法

    bind 函数相当于 es6 的 Function.prototype.bind 方法,只不过为了兼容性,axios 在这里自己实现了一遍

  • utils.extend(instance, Axios.prototype, context);
    

    这一步,通过 extend 方法,将 Axios.prototype ,也就是基类的原型链,合并(继承)到 instancerequest) 对象上,这之中当然是冗余的,也就是 instance 的原型上也包含自身 request 方法,到这里就实现了 axios(config)(相当于 axios.request(config))以及 axios.get / axios.post ... (原型链方法)这几个功能

    这里 extend 函数的第三个参数是 context,作用是将 context 对象当作 Axios.prototype 函数的上下文 this,extend 也是自己实现的方法,感兴趣的同学可以自行查看

  • utils.extend(instance, context)
    

    这一步,还是通过 extend 方法,将 context(实例)的其他属性,例如:defaults、interceptors 合并到 instance 对象上

  • axios.create = function create(instanceConfig) {
      return createInstance(mergeConfig(axios.defaults, instanceConfig));
    };
    

    这里将 createInstance 函数扩展为 axios.create ,允许开发者生成新的实例,同样的,也扩展了 Cancelall 等方法

最后,我们得到了一个对象 axios ,它看起来像 Axios 类的实例,用起来也像实例,然而它只是一个 request 方法,但是拥有实例的属性和原型,以及一些其他的 API

年少的我曾经想当然的认为,通过 axios.create 创建的实例,还可以继续通过 create 创建子类,现在看了源码之后终于悟了,原来根本就没有这层设计,原来我站在第五层,而作者才到第二层(滑稽)

axios 的默认配置

接下来我们在深入具体逻辑之前,先看一下,Axios 都有什么默认配置,如何实现修改全局配置,以及区分实例的配置

再回看一下上面的代码以及 Axios 基类

// lib/core/Axios.js
function Axios(instanceConfig) {
  this.defaults = instanceConfig;
  this.interceptors = {
    request: new InterceptorManager(),
    response: new InterceptorManager()
  };
}
// lib/axios.js
var defaults = require('./defaults');
// Create the default instance to be exported
var axios = createInstance(defaults);

// Factory for creating new instances
axios.create = function create(instanceConfig) {
  return createInstance(mergeConfig(axios.defaults, instanceConfig));
};

全局配置是一个文件,通过 new 一个实例,绑定到 axios 对象上的 defaults 属性上,而之后通过 axios.create 创建的实例,都要和 axios.defaults 合并之后再实例化,因此,axios.defaluts 就变成了一个全局配置,修改该属性会影响 axios 对象以及通过它派生出来的所有实例,而实例上有自己的 defaults 对象,修改它只会影响自己

接下来我们看一下具体的默认配置都有哪些

var DEFAULT_CONTENT_TYPE = {
  'Content-Type': 'application/x-www-form-urlencoded'
};

function getDefaultAdapter() {
  var adapter;
  if (typeof XMLHttpRequest !== 'undefined') {
    // For browsers use XHR adapter
    adapter = require('./adapters/xhr');
  } else if (typeof process !== 'undefined' && Object.prototype.toString.call(process) === '[object process]') {
    // For node use HTTP adapter
    adapter = require('./adapters/http');
  }
  return adapter;
}

var defaults = {
  adapter: getDefaultAdapter(),
  transformRequest: [...],
  transformResponse: [...],
  timeout: 0,
  xsrfCookieName: 'XSRF-TOKEN',
  xsrfHeaderName: 'X-XSRF-TOKEN',
  maxContentLength: -1,
  maxBodyLength: -1,
  validateStatus: function validateStatus(status) {
    return status >= 200 && status < 300;
  }
};
defaults.headers = {
  common: {
    'Accept': 'application/json, text/plain, */*'
  }
};
utils.forEach(['delete', 'get', 'head'], function forEachMethodNoData(method) {
  defaults.headers[method] = {};
});
utils.forEach(['post', 'put', 'patch'], function forEachMethodWithData(method) {
  defaults.headers[method] = utils.merge(DEFAULT_CONTENT_TYPE);
});
module.exports = defaults;

这里省略了一些细节,具体可以去源代码里查看,但是看到这里是不是很多同学反应过来了,原来文档里说的根本不是默认配置,只是列举了一下所有可用的配置以及应该传什么值,而真正的默认配置只有上面文件里的这些

  • adapter

    这里区分了浏览器环境还是 Node 环境,这里分别匹配了 lib/adapters/xhr.js 和 lib/adapters/http.js 文件,也允许开发者自己做适配,反正我还没有适配过...

  • transformRequest 和 transformResponse

    这里都有默认配置,判断分支还挺多,如果修改配置的话,直接赋值,这些逻辑就都没了,个人感觉这里设计的不是很好,所以如果只是想增加新的 transform 规则的话,我建议在默认的基础上新增:

    axios.defaults.transformRequest = [ ...axios.defaults.transformRequest, ...customerTransform ]

  • timeout、xsrfCookieName、xsrfHeaderName、maxContentLength、maxBodyLength、validateStatus

    这些配置都很容易理解,这里不赘述

  • headers

    默认的 header 头,有一个通用 Accept 头,然后具体区分了一下请求类型,只有 'post', 'put', 'patch' 这三种请求会带上默认的 'Content-Type': 'application/x-www-form-urlencoded'看清楚了啊,可不是 'application/json' (狗头)

总结

第一部分暂时就写到这里,主要介绍了 Axios 的实例和默认配置以及,其实源码并不复杂,只是做了比较详细的功能拆分,接下来第二部分我会深入 request 请求的核心,以及拦截器的实现,作为搬砖人,能力有限,如果有童鞋对这一部分的哪里还没有看懂,觉得我没有描述清楚的,可以在留言区讨论