当前位置: 首页 > news >正文

axios执行原理了解一下!

axios 源码解析

axios是一个基于Promise的http请求库,可用于浏览器和 Node。可以说是目前最为常用的http库,有必要了解一下其内部的实现原理

以一个简单示例进行说明

const axios = require('axios');
axios.defaults.baseURL = 'http://xxx.com/api';
axios.interceptors.request.use(resolveFn1, rejectFn2); // 添加请求拦截器
axios.interceptors.response.use(resolveFn2, rejectFn2); // 添加响应拦截器
axios.get('/get').then(() => {
    // 请求成功的处理
  }, () => {
    // 请求异常的处理
  }
);
复制代码

上述代码演示了如何发起axios请求,先从require('axios')说起。 require('axios')导出值的来自./lib/axios.js,而./lib/axios.js导出是内部调用createInstance之后的返回值。createInstance方法会返回一个axios实例(注:axios.create也可以创建axios实例,其内部也是调用createInstance)。我们先来看下createInstance的源码是如何实现的

createInstance

// ./lib/axios.js
function createInstance(defaultConfig) {
  // 根据默认设置 新建一个Axios对象
  var context = new Axios(defaultConfig);

  // axios中所有的请求[axios, axios.get, axios.post等...]内部调用的都是Axios.prototype.request,见[./code/Axios.js]
  // 将Axios.prototype.request的内部this绑定到新建的Axios对象上,从而形成一个axios实例
  var instance = bind(Axios.prototype.request, context);

  utils.extend(instance, Axios.prototype, context); // 将Axios.prototype属性添加到instance上,如果属性为函数则绑定this为context后再添加

  utils.extend(instance, context); // 将新建的Axios对象属性添加到instance,同上

  return instance;
}
复制代码

首先内部会新建一个Axios对象,Axios结构函数如下

function Axios(instanceConfig) {
  this.defaults = instanceConfig;        // 一些默认设置项
  this.interceptors = {
    request: new InterceptorManager(),   // request拦截器
    response: new InterceptorManager()   // response拦截器
  };
}
复制代码

新建的Axios对象主要是用来挂载axios实例的一些设置(如defaults会挂载axios实例的通用设置,interceptors用于存放拦截器)

根据源码可知,axios实例(instance)是对Axios.prototype.request方法包裹了一层函数,主要是为将Axios.prototype.request内部的this绑定到新建的Axios对象上。然后通过 utils.extend 将内部contextAxios.prototyp的属性添加到这个Axios.prototype.request方法上,添加上去的函数也会绑定this到新建的Axios对象上。最终的axios实例上面的方法内部的this指向的都是新建的Axios对象,从而使得不同axios实例之间隔离了作用域,可以对每个axios实例设置不同的config

为什么不将所有方法在Axios上实现然后返回new Axios呢?

因为axios内部调用的都是Axios.prototype.request方法,Axios.prototype.request默认请求方法为get。为了让开发者可以直接axios(config)就可以发送get请求,而不需要axios.get(config)。如果直接new一个Axios对象是无法实现这种简写的(没错,就是为了少打几个字)

实际上axios.post、axios.put等所有axios的请求方法内部都是调用Axios.prototype.request

Axios.prototype.request

// 见./lib/core/Axios
Axios.prototype.request = function request(config) {
  if (typeof config === 'string') {
    config = utils.merge({
      url: arguments[0]
    }, arguments[1]);
  }
  // 进行配置项的合并  优先级: Axios默认的defaults < Axios.defaults < 调用时axios请求方法时传入的config
  config = utils.merge(defaults, {
    method: 'get'               // 默认为get方法
  }, this.defaults, config);
  config.method = config.method.toLowerCase();

  var chain = [dispatchRequest, undefined]; // dispatchRequest封装了对于发起ajax的逻辑处理
  var promise = Promise.resolve(config);

  // request拦截器的执行顺序是: 先加入后执行
  this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {
    chain.unshift(interceptor.fulfilled, interceptor.rejected);
  });

  // 而response拦截器则是: 先加入的先执行
  this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {
    chain.push(interceptor.fulfilled, interceptor.rejected);
  });

  /*
    假如我们分别添加了2个request和respnse拦截器, 那么最终执行顺序如下:
    request.interceptor2 => request.interceptor1 => [dispatchRequest, undefined] => response.interceptor1 => response.interceptor2
    内部通过promise.then形成promise链, 从而将chain中拦截器的调用串联起来, dispatchRequest是对于ajax请求发起的封装实现,也会返回一个Promise对象
  */
  while (chain.length) {
    promise = promise.then(chain.shift(), chain.shift());
  }

  return promise;
};
复制代码

Axios.protytype.request内部会进行了一些配置项的合并工作,变量chain相当于一个任务队列,以2个为一组存放任务(1个是任务成功回调,1个是任务失败回调),通过不断调用promise.then方法形成一个promise链,从而将所有的任务执行串联起来。

有一点需要注意是拦截器的执行顺序,request 拦截器先加入的后执行,response 拦截器则是先加入的先执行。

 执行顺序示例:request.interceptor2 => request.interceptor1 => [dispatchRequest, undefined] => response.interceptor1 => response.interceptor2
复制代码

request.interceptor用于请求发起前的准备工作(可以修改data和headers),response.interceptor用于服务器返回数据之后的处理工作(也是对data进行处理),整个请求过程的发起过程是通过dispatchRequest实现的

dispatchRequest

// 省略部分代码,详细代码见./lib/code/dispatchRequest
function dispatchRequest(config) {
  // ...
  // 依次调用transformRequest数组中的函数对data,headers进行处理,方便在向服务器发送请求之前对data和headers进行修改(例如对data进行编码加密等)
  config.data = transformData(
    config.data,
    config.headers,
    config.transformRequest
  );
  // ...
  return adapter(config).then(
    function onAdapterResolution(response) {
      // 判断请求是否被取消,如果请求已经被手动取消则会抛出一个异常
      throwIfCancellationRequested(config);

      // Transform response data
      // 利用transformResponse对服务器返回的data进行处理
      response.data = transformData(response.data, response.headers, config.transformResponse);

      return response;
    },
    function onAdapterRejection(reason) {
      // 执行到这里说明请求出现了异常(代码执行出错或者状态码错误等),但是如果这是执行取消请求操作,那么最终的异常信息还是取消请求所抛出的异常,这样是为了当开发者手动取消请求时,可以对所有取消请求进行统一的后续处理
      if (!isCancel(reason)) {
      throwIfCancellationRequested(config);  

      // Transform response data
      if (reason && reason.response) {
        reason.response.data = transformData(
          reason.response.data,
          reason.response.headers,
          config.transformResponse
        );
      }
    }
    
    return Promise.reject(reason);
    }
  );
}
复制代码

transformData(config.data, config.headers, config.transformRequest)是为了向服务器发送请前对 data 进行处理,可以通过设置transformRequest对data和header进行修改,一般进行加密编码等操作。

adapter 是一个典型的适配器模式的实现, 其默认值为getDefaultAdapter的返回值

// 见./lib/cord/defaults.js
// 根据当前执行环境(浏览器 or Node)执行相应的请求发起逻辑
function getDefaultAdapter() {
  var adapter;
  if (typeof XMLHttpRequest !== 'undefined') { // 浏览器环境
    // For browsers use XHR adapter
    adapter = require('./adapters/xhr');
  } else if (typeof process !== 'undefined') { // node环境
    // For node use HTTP adapter
    adapter = require('./adapters/http');
  }
  return adapter;
}
复制代码

可以看到adapter内部对不同环境做了适配处理,封装了统一的行为: 根据config发送请求然后返回一个promise,promise的状态根据请求的结果来决定。 各个环境具体的实现,可自行阅读源码

接下来我们来看下adapter返回promise的成功和失败回调是如何处理的

  1. onAdapterResolution方法
  • 调用throwIfCancellationRequested来判断请求是否被取消(axios中可以通过cancelToken取消请求),如果请求已经被手动取消则会抛出一个异常
  • 调用transformResponse对服务返回的数据进行处理,一般进行解密解码等操作
  • 返回处理之后的response给开发者使用
  1. onAdapterRejection
  • 请求失败的原因可分为2种,1种是普通的请求异常(如:后台返回了错误的状态码、代码执行出错等),另一种是我们手动取消了请求从而抛出的异常,2者可以根据isCancel进行区分。注意:当执行过程同时出现2种异常时,axios返回的异常最终会是取消请求所抛的异常

至此 axios 库的处理流程就结束了。

结语

本文大致讲了一下axios的执行原理,还有很多细节没有涉及,更多见源码注释
由于之前没有使用过axios╮(╯▽╰)╭ ,如有不对之处请多多指教!!!

相关文章:

  • MySQL语句执行分析(二)
  • 大数据争论:批处理与流处理的C位之战
  • 汉诺(hanio)塔问题
  • docker 系列 - Docker CheatSheet | Docker 配置与实践清单 (转载)
  • CentOS下rpm指令和yum指令详解
  • 微软产品大升级:Surface Pro 6、Studio 2、Laptop 2 重磅来袭
  • mysql8.0 Authentication plugin 'caching_sha2_password' cannot be loaded
  • PostgreSQL 函数式索引使用注意 - 暨非immutable函数不适合索引的原因
  • 零基础兴趣或者转行学习Python,我们应该如何入门呢?
  • bartender 9.4 错误消息6670 无法链接到数据库的解决办法
  • JVM G1笔记
  • Linux下切换用户出现su: Authentication failure的解决办法
  • [MicroPython]TPYBoard v102 CAN总线通信
  • Java多线程——AQS框架源码阅读
  • 超大数据下大批量随机键值的查询优化方案
  • 分享的文章《人生如棋》
  • android百种动画侧滑库、步骤视图、TextView效果、社交、搜房、K线图等源码
  • Django 博客开发教程 8 - 博客文章详情页
  • docker python 配置
  • es6
  • isset在php5.6-和php7.0+的一些差异
  • Js基础知识(四) - js运行原理与机制
  • Less 日常用法
  • supervisor 永不挂掉的进程 安装以及使用
  • vue.js框架原理浅析
  • 紧急通知:《观止-微软》请在经管柜购买!
  • 如何使用Mybatis第三方插件--PageHelper实现分页操作
  • 一起参Ember.js讨论、问答社区。
  • 原生Ajax
  • 好程序员web前端教程分享CSS不同元素margin的计算 ...
  • 回归生活:清理微信公众号
  • 京东物流联手山西图灵打造智能供应链,让阅读更有趣 ...
  • ​ArcGIS Pro 如何批量删除字段
  • ​无人机石油管道巡检方案新亮点:灵活准确又高效
  • (3)选择元素——(17)练习(Exercises)
  • (安卓)跳转应用市场APP详情页的方式
  • (八)c52学习之旅-中断实验
  • (附源码)ssm基于jsp的在线点餐系统 毕业设计 111016
  • (四)库存超卖案例实战——优化redis分布式锁
  • (心得)获取一个数二进制序列中所有的偶数位和奇数位, 分别输出二进制序列。
  • (转)Unity3DUnity3D在android下调试
  • .NET CF命令行调试器MDbg入门(一)
  • .NET 线程 Thread 进程 Process、线程池 pool、Invoke、begininvoke、异步回调
  • .NET/C# 使用反射注册事件
  • .NET6 命令行启动及发布单个Exe文件
  • .NET教程 - 字符串 编码 正则表达式(String Encoding Regular Express)
  • .pyc文件是什么?
  • .w文件怎么转成html文件,使用pandoc进行Word与Markdown文件转化
  • @Bean有哪些属性
  • @vue/cli脚手架
  • [ vulhub漏洞复现篇 ] ThinkPHP 5.0.23-Rce
  • [.NET 即时通信SignalR] 认识SignalR (一)
  • []error LNK2001: unresolved external symbol _m
  • []新浪博客如何插入代码(其他博客应该也可以)
  • [AI]文心一言爆火的同时,ChatGPT带来了这么多的开源项目你了解吗