引言
熟悉 Vue 的同学们,肯定都用过 Axios , 简洁的 API 语法和方便的拦截器都受到了开发者的热烈欢迎,使用背后,是否也想了解一下实现原理呢,其实源码并不难阅读,这里我就抛砖引玉,对源码做一个浅析
回顾
首先我们回顾一下常用的 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 是更加通用的方法,而这个目录里的则是定制的一些辅助方法
源码分析
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
函数将基类Axios
的request
方法的上下文this
绑定为实例context
,并生成一个新的方法instance
,也就是未来我们用的axios
对象,到这一步,axios
对象本质上还只是一个request
方法 -
utils.extend(instance, Axios.prototype, context);
这一步,通过 extend 方法,将
Axios.prototype
,也就是基类的原型链,合并(继承)到instance
(request
) 对象上,这之中当然是冗余的,也就是instance
的原型上也包含自身request
方法,到这里就实现了axios(config)
(相当于axios.request(config)
)以及axios.get / axios.post ...
(原型链方法)这几个功能 -
utils.extend(instance, context)
这一步,还是通过 extend 方法,将
context
(实例)的其他属性,例如:defaults、interceptors 合并到instance
对象上 -
axios.create = function create(instanceConfig) { return createInstance(mergeConfig(axios.defaults, instanceConfig)); };
这里将
createInstance
函数扩展为axios.create
,允许开发者生成新的实例,同样的,也扩展了Cancel
,all
等方法
最后,我们得到了一个对象 axios
,它看起来像 Axios
类的实例,用起来也像实例,然而它只是一个 request
方法,但是拥有实例的属性和原型,以及一些其他的 API
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 请求的核心,以及拦截器的实现,作为搬砖人,能力有限,如果有童鞋对这一部分的哪里还没有看懂,觉得我没有描述清楚的,可以在留言区讨论
常见问题FAQ
- 免费下载或者VIP会员专享资源能否直接商用?
- 本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
- 提示下载完但解压或打开不了?
- 找不到素材资源介绍文章里的示例图片?
- 模板不会安装或需要功能定制以及二次开发?
发表评论
还没有评论,快来抢沙发吧!