经过上文的学习,已经对axios源码有了一些了解,知道了axios对象中包含哪些属性和方法,以及最终导出的axios其实是一个函数。但是对其中实例的原理,以及各种方法的原理却没有深入,从这一篇文章就开始对这些内容的学习。
在这个文件中,主要内容可分为五个部分:
// 工具函数
var utils = require('./../utils');
// 辅助函数
var buildURL = require('../helpers/buildURL');
// 拦截器对象
var InterceptorManager = require('./InterceptorManager');
// 派发请求方法
var dispatchRequest = require('./dispatchRequest');
// 合并配置
var mergeConfig = require('./mergeConfig');
// 校验方法
var validator = require('../helpers/validator');
在上一文章中,我们有提到够Axios实例的内容:defaults属性和interceptors属性,这两个属性的具体内容如下:
function Axios(instanceConfig) {
this.defaults = instanceConfig;
this.interceptors = {
request: new InterceptorManager(),
response: new InterceptorManager()
};
}
Axios.prototype.getUri = function getUri(config) {
config = mergeConfig(this.defaults, config);
return buildURL(config.url, config.params, config.paramsSerializer).replace(/^?/, '');
};
mergeConfig方法用于合并默认配置和使用时传入的配置项。这部分的主要内容是在mergeConfig.js的后半部分
module.exports = function mergeConfig(config1, config2) {
config2 = config2 || {};
var config = {};
var mergeMap = {
'url': valueFromConfig2,
'method': valueFromConfig2,
'data': valueFromConfig2,
'baseURL': defaultToConfig2,
'transformRequest': defaultToConfig2,
'transformResponse': defaultToConfig2,
'paramsSerializer': defaultToConfig2,
'timeout': defaultToConfig2,
'timeoutMessage': defaultToConfig2,
'withCredentials': defaultToConfig2,
'adapter': defaultToConfig2,
'responseType': defaultToConfig2,
'xsrfCookieName': defaultToConfig2,
'xsrfHeaderName': defaultToConfig2,
'onUploadProgress': defaultToConfig2,
'onDownloadProgress': defaultToConfig2,
'decompress': defaultToConfig2,
'maxContentLength': defaultToConfig2,
'maxBodyLength': defaultToConfig2,
'transport': defaultToConfig2,
'httpAgent': defaultToConfig2,
'httpsAgent': defaultToConfig2,
'cancelToken': defaultToConfig2,
'socketPath': defaultToConfig2,
'responseEncoding': defaultToConfig2,
'validateStatus': mergeDirectKeys
};
utils.forEach(Object.keys(config1).concat(Object.keys(config2)), function computeConfigValue(prop) {
var merge = mergeMap[prop] || mergeDeepProperties;
var configValue = merge(prop);
(utils.isUndefined(configValue) && merge !== mergeDirectKeys) || (config[prop] = configValue);
});
return config;
}
bindURL方法用于对请求参数做处理,根据不同的参数形式去做不同的处理
如果没有params参数,则直接返回url
如果自带参数解析方法paramsSerializer,则通过paramsSerializer进行解析处理
如果params是URLSearchParams格式,则将参数转为字符串即可
如果不符合以上判断条件,则进行axios自己的参数处理操作
如果参数为空,直接返回
如果参数是数组类型,则在属性名后面加一个字符串’[]’,反之将属性值转为数组形式
对属性值进行遍历
最终将连接的呢绒再以&符号连接
如果存在参数解析的值,则要对URL进行处理,去除掉哈希模式下的#号,再将参数通过?拼接上去
module.exports = function buildURL(url, params, paramsSerializer) {
if (!params) {
return url;
}
var serializedParams;
if (paramsSerializer) {
serializedParams = paramsSerializer(params);
} else if (utils.isURLSearchParams(params)) {
serializedParams = params.toString();
} else {
var parts = [];
utils.forEach(params, function serialize(val, key) {
if (val === null || typeof val === 'undefined') {
return;
}
if (utils.isArray(val)) {
key = key + '[]';
} else {
val = [val];
}
utils.forEach(val, function parseValue(v) {
if (utils.isDate(v)) {
v = v.toISOString();
} else if (utils.isObject(v)) {
v = JSON.stringify(v);
}
parts.push(encode(key) + '=' + encode(v));
});
});
serializedParams = parts.join('&');
}
if (serializedParams) {
var hashmarkIndex = url.indexOf('#');
if (hashmarkIndex !== -1) {
url = url.slice(0, hashmarkIndex);
}
url += (url.indexOf('?') === -1 ? '?' : '&') + serializedParams;
}
return url;
};
方法的代码太长,分作几部分介绍:
if (typeof config === 'string') {
config = arguments[1] || {};
config.url = arguments[0];
} else {
config = config || {};
}
config = mergeConfig(this.defaults, config);
这样处理的原因在于传入配置参数的不同。
还记得在第一篇文章中介绍axios使用方式时的API方式吗?里面分为了两种使用方式。
axios({
method: 'get',
url: 'xxx',
data: {}
});
axios('http://xxx');
axios('http://xxx',{
xxx
});
这里的第一个判断条件就是axios(url,config)的情况,第二个是axios(config)的情况,最终将传入的配置和默认配置项进行合并。
if (config.method) {
config.method = config.method.toLowerCase();
} else if (this.defaults.method) {
config.method = this.defaults.method.toLowerCase();
} else {
config.method = 'get';
}
由于前端传入时,请求方式大小写形式都有,所以要统一处理成小写形式,以及设置了默认的请求方式
var transitional = config.transitional;
if (transitional !== undefined) {
validator.assertOptions(transitional, {
silentJSONParsing: validators.transitional(validators.boolean),
forcedJSONParsing: validators.transitional(validators.boolean),
clarifyTimeoutError: validators.transitional(validators.boolean)
}, false);
}
这段代码是对transitional 属性进行版本校验,会提示自某个版本之后移除该属性。
我在网上看到过validators.boolean会跟一个’1.0.0’版本号,但是我从gitee网站上clone的最新代码是没有,不晓得是哪里的改变了。
这一部分,我也是看不太懂。在网上找到了仙凌阁大大的文章https://blog.csdn.net/qq_39221436/article/details/120652086
复制了拦截器部分的解读代码,由于这部分代码经过一次重构,所以两种代码的展示都放到下面了。
重构前的代码内容不多,看起来也不是很难以理解,更便于初学者学习。
// 创建存储链式调用的数组 首位是核心调用方法dispatchRequest,第二位是空
var chain = [dispatchRequest, undefined];
// 创建 promise 为什么resolve(config)是因为 请求拦截器最先执行,
// 所以设置请求拦截器时可以拿到每次请求的所有config配置
var promise = Promise.resolve(config);
// 把设置的请求拦截器的成功处理函数、失败处理函数放到数组最前面
this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {
chain.unshift(interceptor.fulfilled, interceptor.rejected);
});
// 把设置的响应拦截器的成功处理函数、失败处理函数放到数组最后面
this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {
chain.push(interceptor.fulfilled, interceptor.rejected);
});
// 循环 每次取两个出来组成promise链.then执行
while (chain.length) {
promise = promise.then(chain.shift(), chain.shift());
}
// 返回promise
return promise;
promise链是从左到右执行的,先执行请求拦截器,再执行请求,最后执行响应拦截器。
由于拦截器可以有多个(所以用forEach),所以会有顺序问题,由于unshift和push的方法
主要是为了解决请求拦截器中可能会出现异步情况或当前宏任务执行时间过长而阻塞真正请求执行的问题
// 请求拦截器储存数组
var requestInterceptorChain = [];
// 默认所有请求拦截器都为同步
var synchronousRequestInterceptors = true;
// 遍历注册好的请求拦截器数组
this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {
// 这里interceptor是注册的每一个拦截器对象,axios请求拦截器向外暴露了runWhen配置
// 来针对一些需要运行时检测来执行的拦截器
// 如果配置了该函数,并且返回结果为true,则记录到拦截器链中,反之则直接结束该层循环
if (typeof interceptor.runWhen === 'function' && interceptor.runWhen(config) === false) {
return;
}
// interceptor.synchronous 是对外提供的配置,可标识该拦截器是异步还是同步 默认为false(异步)
// 这里是来同步整个执行链的执行方式的,如果有一个请求拦截器为异步,
// 那么下面的promise执行链则会有不同的执行方式
synchronousRequestInterceptors = synchronousRequestInterceptors && interceptor.synchronous;
// 塞到请求拦截器数组中
requestInterceptorChain.unshift(interceptor.fulfilled, interceptor.rejected);
});
// 相应拦截器存储数组
var responseInterceptorChain = [];
// 遍历按序push到拦截器存储数组中
this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {
responseInterceptorChain.push(interceptor.fulfilled, interceptor.rejected);
});
var promise;
// 如果为异步 其实也是默认情况
if (!synchronousRequestInterceptors) {
// 这里和重构之前的逻辑是一致的了
var chain = [dispatchRequest, undefined];
// 请求拦截器塞到前面
Array.prototype.unshift.apply(chain, requestInterceptorChain);
// 响应拦截器塞到后面
chain = chain.concat(responseInterceptorChain);
promise = Promise.resolve(config);
// 循环 执行
while (chain.length) {
promise = promise.then(chain.shift(), chain.shift());
}
// 返回promise
return promise;
}
// 这里则是同步的逻辑
var newConfig = config;
// 请求拦截器一个一个的走 返回 请求前最新的config
while (requestInterceptorChain.length) {
var onFulfilled = requestInterceptorChain.shift();
var onRejected = requestInterceptorChain.shift();
// 做异常捕获 有错直接抛出
try {
newConfig = onFulfilled(newConfig);
} catch (error) {
onRejected(error);
break;
}
}
// 到这里 微任务不会过早的创建 也就解决了微任务过早创建、
// 当前宏任务过长或某个请求拦截器中有异步任务而阻塞真正的请求延时发起问题
try {
promise = dispatchRequest(newConfig);
} catch (error) {
return Promise.reject(error);
}
// 响应拦截器执行
while (responseInterceptorChain.length) {
promise = promise.then(responseInterceptorChain.shift(), responseInterceptorChain.shift());
}
return promise;
设置别名的方式更便于axios的使用,在第一篇文章中,我们也列举了别名方式的使用示例。
而在设置别名的时候,由于有些方式是没有data参数的,所以也需要分类处理。
utils.forEach(['delete', 'get', 'head', 'options'],function forEachMethodNoData(method) {
Axios.prototype[method] = function(url, config) {
return this.request(mergeConfig(config || {}, {
method: method,
url: url,
data: (config || {}).data
}));
};
});
utils.forEach(['post', 'put', 'patch'],function forEachMethodWithData(method) {
Axios.prototype[method] = function(url, data, config) {
return this.request(mergeConfig(config || {}, {
method: method,
url: url,
data: data
}));
};
});
module.exports = Axios;