Axios源码阅读(二):拦截器

一、拦截器介绍

先看下官方文档对拦截器的介绍:

You can intercept requests or responses before they are handled by then or catch.

即我们可以在axios返回的promise定型(resolve)之前拦截请求和响应。这里是文档的示例代码:

// Add a request interceptor
axios.interceptors.request.use(function (config) {
    // Do something before request is sent
    return config;
  }, function (error) {
    // Do something with request error
    return Promise.reject(error);
  });

// Add a response interceptor
axios.interceptors.response.use(function (response) {
    // Any status code that lie within the range of 2xx cause this function to trigger
    // Do something with response data
    return response;
  }, function (error) {
    // Any status codes that falls outside the range of 2xx cause this function to trigger
    // Do something with response error
    return Promise.reject(error);
  });

// If you need to remove an interceptor later you can.
const myInterceptor = axios.interceptors.request.use(function () {/*...*/});
axios.interceptors.request.eject(myInterceptor);

还有些功能示例代码并没有告诉我们,我们去github上看下详细的拦截器文档,看完之后,我对这部分功能点做了个总结:

  1. Axios可以添加请求、响应拦截器;
  2. 请求拦截器可以在请求被发出之前做一些处理;
  3. 响应拦截器可以在用户代码获取到响应之前对响应做一些处理;
  4. Axios可以移除拦截器;
  5. 用户可以决定请求拦截器是同步执行还是异步执行;
  6. 用户可以决定拦截器是否执行。

下面,我们将围绕以上6个功能进行源码阅读。

二、拦截器源码阅读

2.1 添加拦截器

文档指出我们可以通过axios.interceptors.request.useaxios.interceptors.response.use来添加拦截器,那么我们一步一步看这个功能是如何实现的。

axios.interceptors

axios上扩展了Axios实例的属性,所以我们在Axios.js中找到了axios.interceptors的来源:

传送门:./lib/core/Axios.js

/**
 * Create a new instance of Axios
 *
 * @param {Object} instanceConfig The default config for the instance
 */
function Axios(instanceConfig) {
  this.defaults = instanceConfig;
  this.interceptors = {
    request: new InterceptorManager(),
    response: new InterceptorManager()
  };
}

所以axios.interceptors里面是两个拦截器管理对象

InterceptorManager

我们直接找到拦截器管理器的源码:

传送门:./lib/core/InterceptorManager.js

function InterceptorManager() {
  this.handlers = [];
}

/**
 * Add a new interceptor to the stack
 *
 * @param {Function} fulfilled The function to handle `then` for a `Promise`
 * @param {Function} rejected The function to handle `reject` for a `Promise`
 *
 * @return {Number} An ID used to remove interceptor later
 */
InterceptorManager.prototype.use = function use(fulfilled, rejected, options) {
  this.handlers.push({
    fulfilled: fulfilled,
    rejected: rejected,
    synchronous: options ? options.synchronous : false,
    runWhen: options ? options.runWhen : null
  });
  return this.handlers.length - 1;
};

InterceptorManager内部只有一个实例属性handlers,引用的是一个空数组;InterceptorManager.prototype.use接受三个参数,分别是fulfilled callbackrejected callbackoptions,然后把这三个参数包装为一个对象pushhandlers中,并返回在handlers中的索引。这便是添加拦截器的过程。

2.2 拦截器的工作原理

通过上篇文章:Axios源码阅读(一):核心功能源码,我们知道Axios.prototype.request中描述了拦截器是如何工作的:

传送门:./lib/core/Axios.js

// filter out skipped interceptors
  var requestInterceptorChain = [];
  var synchronousRequestInterceptors = true;
  
  this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {
    // 对每一个请求拦截器执行以下代码
    if (typeof interceptor.runWhen === 'function' && interceptor.runWhen(config) === false) {
      // 如果 runWhen 的结果是 false ,那么不执行这个拦截器
      return;
    }
    
    // 默认拦截器是异步的,并且只有所有的拦截器都是同步的,才会同步执行
    synchronousRequestInterceptors = synchronousRequestInterceptors && interceptor.synchronous;
    
    requestInterceptorChain.unshift(interceptor.fulfilled, interceptor.rejected);
  });

  var responseInterceptorChain = [];
  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());
    }

    return promise;
  }

  // 如果拦截器是同步执行
  var newConfig = 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);
  }
  
  // 把所有的响应拦截器追加到 dispatchRequest 返回的 promise 后面
  while (responseInterceptorChain.length) {
    promise = promise.then(responseInterceptorChain.shift(), responseInterceptorChain.shift());
  }

  return promise;

小结

在这里总结下拦截器的执行步骤:

  1. 发起请求前声明请求拦截器数组和响应拦截器数组;
  2. 遍历axios实例上的请求拦截器,执行runWhen(如果存在的话),将结果为true的拦截器的callback压入请求拦截器数组中,并判断拦截器是同步还是异步;
  3. 将所有的响应拦截器的callback压入响应拦截器数组中;
  4. 如果所有的,注意是所有的请求拦截器都是同步的(将在下文论证这个观点),那么就先循环执行完所有的请求拦截器,然后返回从dispatchRequest开始的promise;否则,所有的请求拦截器都是异步执行,即返回第一个拦截器执行的promise

2.3 移除拦截器

示例代码中通过axios.interceptors.request.eject可以移除拦截器,我们找到相应代码:

传送门:./lib/core/InterceptorManager.js

/**
 * Remove an interceptor from the stack
 *
 * @param {Number} id The ID that was returned by `use`
 */
InterceptorManager.prototype.eject = function eject(id) {
  if (this.handlers[id]) {
    this.handlers[id] = null;
  }
};

这里的实现很简单,通过将handlers数组中对应的拦截器置为null就可以了,那这个索引是从哪里来的呢?其实来源于use方法会返回拦截器在handlers数组中的索引。

2.4 拦截器的同步执行和条件执行

同步执行

InterceptorManager.prototype.use接收的第三个参数为一个对象,在该对象上可以配置拦截器是同步还是异步执行,但是每个拦截器都有这个配置项,如果配置项不同的话,会产生什么后果呢?比如有的拦截器配置的是同步执行,有的拦截器配置的是异步执行,那拦截器最终是如何执行的呢?

通过源码,我们不难发现只有在所有的拦截器都是同步执行的情况下,拦截器才会同步执行,否则会异步执行。这里我们实验论证下:

axios.interceptors.request.use(config => {
    console.log('请求拦截器1');
    return config;
}, error => {}, {
    synchronous: true
});

axios.interceptors.request.use(config => {
    console.log('请求拦截器2');
    return config;
}, error => {}, {
    synchronous: false
});

axios.get('/api').then(res => {
    console.log(res);
})

console.log('同步代码执行');

执行结果:


异步执行
axios.interceptors.request.use(config => {
    console.log('请求拦截器1');
    return config;
}, error => {}, {
    synchronous: true
});

axios.interceptors.request.use(config => {
    console.log('请求拦截器2');
    return config;
}, error => {}, {
    synchronous: true
});

console.log('同步代码执行');

执行结果:


同步执行

通过以上实验可以得出:只有当所有拦截器都是同步执行时,拦截器才会同步执行,否则都会异步执行。

条件执行

当拦截器需要条件执行的时候,只需要在use的时候配置好runWhen就行了

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

推荐阅读更多精彩内容