axios核心模块原理剖析

axios是一个基于promise调用逻辑的http请求库,是一个优秀的开源项目。了解其实现逻辑有助于深化我们对接口往返的理解,提高Promise的应用能力等。本文会挑几个axios重点且常用的功能模块进行原理剖析并使用ts实现,完整代码在这里。本文对每个模块将按照功能描述,原理剖析,ts实现的顺序进行。

请求参数与url的处理

axios会对我们传入的请求参数做统一处理,当发送的是get请求时会根据不同数据类型做不同的拼接处理,总结有以下几点

  • 基本类型 按基础规则拼接

    axios({
      method: 'get',
      url: '/test/get',
      params: {
        a: 1,
        b: 2,
      },
    });
    // http://localhost:8080/simple/get?a=1&b=2
    
  • 数组 属性名拼接数组符号并依次拼接各元素

    axios({
      method: 'get',
      url: '/test/get',
      params: {
        arr: [1, 2],
      },
    });
    // http://localhost:3000/test/get?arr[]=1&arr[]=2
    
  • 对象 encode后拼接

    axios({
      method: 'get',
      url: '/test/get',
      params: {
        obj: {
          name: 'foo',
        },
      },
    });
    // http://localhost:3000/test/get?obj=%7B%22name%22:%22foo%22%7D
    
  • Date toString后拼接

    axios({
      method: 'get',
      url: '/test/get',
      params: {
        date: new Date(),
      },
    });
    // http://localhost:3000/test/get?date=2021-11-02T07:58:49.323Z
    
  • 特殊字符不被encode @:$,[],允许他们存在url中。

  • 忽略空值 类型为null或者undefined的值不会被添加到url中。

  • 忽略哈希标记#.

  • 保存已存在的url参数,传入的参数会继续拼接在已存在的参数后。

接下来具体实现,命名为buildUrl

//  声明两个工具函数 用于判断Date类型和object类型
function isDate(val: any) {
  return toString.call(val) === '[object Date]'
}

function isPlainObject(val: any) {
  return toString.call(val) === '[object Object]'
}
function buildURL(url: string, params?: any): string {
  if (!params) {
    return url;
  }
  // 要拼接的参数数组
  const parts: string[] = [];

  Object.keys(params).forEach(key => {
    let val = params[key];
    if (val === null || typeof val === 'undefined') {
      return;
    }
    let values = [];
    if (Array.isArray(val)) {
      values = val;
      // 属性名加'[]'标记
      key += '[]';
    } else {
      values = [val];
    }
    values.forEach(val => {
      if (isDate(val)) {
        val = val.toISOString();
      } else if (isPlainObject(val)) {
        val = JSON.stringify(val);
      }
      parts.push(`${encode(key)}=${encode(val)}`);
    });
  });

  let serializedParams = parts.join('&');

  if (serializedParams) {
    // 忽略哈希标记
    const markIndex = url.indexOf('#');
    if (markIndex !== -1) {
      url = url.slice(0, markIndex);
    }
    // 保留原有的参数
    url += (url.indexOf('?') === -1 ? '?' : '&') + serializedParams;
  }

  return url;
}

默认配置与配置合并策略

axios有默认配置(axios.defaults),我们可以修改默认配置且其能与我们传入axios的配置合并,下面分析其合并策略。

由于axios的配置的是个复杂对象,因此默认配置和自定义配置的合并也不是简单的对象合并。合并的总体原则是,对于基本类型属性(method,timeout,withCredentials。。。)的合并优先使用自定义属性,没有自定义属性则用默认属性,无默认属性则为null(有些属性没有默认值如urlparamsdata,因为这些属性与当次请求强相关。设置默认值无意义),对于object类型(如headers)属性则需要深度合并,即要进行递归判断。

因此,我们要为每一个属性定制其合并策略,接下来按以下步骤具体实现:

  • 声明一个对象strats用来存储各个属性的合并策略
  const strats = Object.create(null);
  • 分别定义默认策略和深度合并策略

    默认策略

    优先用自定义属性 没有则用默认属性 否则返会null

    const defMerge = (target: any, source: any) => {
      return typeof source !== 'undefined' ? source : typeof target !== 'undefined' ? target : null;
    };
    

    深度合并策略
    基本类型直接合并,对象类型值则判断原属性是否是对象类型,如果是,则递归合并。不是,则用一个空对象与之合并。

    const deepMerge = (...objs: any[]): any => {
      const result = Object.create(null)
      objs.forEach(obj => {
        if (obj) {
          Object.keys(obj).forEach(key => {
            const val = obj[key]
            if (isPlainObject(val)) {
              if (isPlainObject(result[key])) {
                result[key] = deepMerge(result[key], val)
              } else {
                result[key] = deepMerge({}, val)
              }
            } else {
              result[key] = val
            }
          })
        }
      })
      return result
    }
    
  • 为每个属性指定合并策略

由于只有少数属性('headers', 'auth')需要深度合并,因此我们只需将需要深度合并的属性及其合并策略注册到strats中,在执行合并时判断当前属性是否在strats中存在即可,存在则执行其专属的合并策略,不存在则执行默认合并策略。

// 需要深度合并的属性
const stratKeysDeepMerge = ['headers', 'auth'];
// 注册合并策略
stratKeysDeepMerge.forEach(key => {
  strats[key] = deepMergeStrat;
});
  • 执行合并

我们约定config1代表默认配置 config2代表自定义配置。
首先声明一个空对象存储合并结果,遍历自定义配置并执行对应合并策略,会优先使用strats中的合并策略,没有则用默认合并策略。之后遍历默认配置,只有合并结果中不存在该属性时再执行其合并策略。

function mergeConfig(config1: AxiosRequestConfig, config2?: AxiosRequestConfig): AxiosRequestConfig {
  if (!config2) {
    config2 = {};
  }
  const config = Object.create(null);
  // 优先合并自定义配置
  for (let key in config2) {
    mergeField(key);
  }
  // 合并默认配置 只有在没有自定义配置时才使用默认配置
  for (let key in config1) {
    if (!config2[key]) {
      mergeField(key);
    }
  }
  function mergeField(key: string): void {
    // 优先自定义配置合并策略 没有则用默认策略
    const strat = strats[key] || defMerge;
    config[key] = strat(config1[key], config2![key]);
  }
  return config;
}

拦截器

axios的拦截器几乎是项目中必用的一项配置,它可以在请求前/响应后对请求体/响应体做一些处理,先来回顾一下基本用法。

  • 使用use方法注册拦截器,使用类似promise.then,接收两个参数,第一个用来添加我们期望拦截器处理的逻辑,第二个参数用来处理错误。

  • 使用eject方法删除某个拦截器。

  • 拦截器可以添加多个,执行顺序是:请求拦截器先添加的后执行,响应拦截器先添加的先执行。

我们知道axios是基于promise实现的,结合拦截器的执行过程其实不难想到可以用promise的链式调用实现,先来回顾一下链式调用。简单来讲Promise.then方法会返回一个Promise,可以继续调用then方法, 前一个then的回调返回的数据会作为参数传入下一个then的回调。因此我们可以将请求/响应拦截器与请求发送的调用使用promise.then方法串联起来。

基于上述,首先实现一个拦截器管理类

interface ResolvedFn<T = any> {
  (val: T): T | Promise<T>
}

interface RejectedFn {
  (error: any): any
}
interface Interceptor<T> {
  resolved: ResolvedFn<T>
  rejected?: RejectedFn
}

class InterceptorManager<T> {
  private interceptors: Array<Interceptor<T> | null>

  constructor() {
    // 用于存放拦截器
    this.interceptors = []
  }
  // 注册拦截器,返回其索引可用于删除
  use(resolved: ResolvedFn<T>, rejected?: RejectedFn): number {
    this.interceptors.push({
      resolved,
      rejected
    })
    return this.interceptors.length - 1
  }
  // 遍历所有拦截器,将每个拦截器作为传入函数的参数并执行
  // 将来用于将拦截器推入promise的调用链中
  forEach(fn: (interceptor: Interceptor<T>) => void): void {
    this.interceptors.forEach(interceptor => {
      if (interceptor !== null) {
        fn(interceptor)
      }
    })
  }
  // 删除拦截器
  eject(id: number): void {
    if (this.interceptors[id]) {
      this.interceptors[id] = null
    }
  }
}

接下来为Axios类添加interceptors属性,他有两个值分别是请求和响应拦截器

export default class Axios {
  constructor() {
    this.interceptors = {
      request: new InterceptorManager<AxiosRequestConfig>(),
      response: new InterceptorManager<AxiosResponse>()
    }
  }
  ......
}

拦截器拦截的是请求,因此最后需要对发送请求的方法进行处理。具体如下:

  • 声明一个chain数组用于存放promise调用链,并首先将请求发送方法放入。

  • 分别调用请求/响应拦截器的foreach方法,将请求拦截器倒序插入数组前部,将响应拦截器顺序插入数组尾部。

  • 声明一个resolved状态的Promise用于启动链式调用,最后循环chain数组,取出每个拦截器,使用then方法调用即可。

request(url: any, config?: any): AxiosPromise {
    // 其他逻辑
    const chain: PromiseChain[] = [{
      resolved: dispatchRequest,
      rejected: undefined
    }]
    // 将请求拦截器倒序插入数组前部
    this.interceptors.request.forEach(interceptor => {
      chain.unshift(interceptor)
    })
    // 将响应拦截器插入数组尾部
    this.interceptors.response.forEach(interceptor => {
      chain.push(interceptor)
    })
    // 初始化一个reslove状态的promise
    let promise = Promise.resolve(config)
    while (chain.length) {
      // 链式调用
      const { resolved, rejected } = chain.shift()!
      promise = promise.then(resolved, rejected)
    }
    return promise
  }

请求取消

请求取消也是axios在项目中的一个常用功能,一个典型场景就是当接口响应慢且会多次触发时(如点击按钮提交,搜索输入框等),由于每次响应时间不定,因此可能出现后发出的请求比先发出的请求的响应速度快的情况,此时就可以使用请求取消,即判断前一次请求结果未返回时,取消当次请求。

回顾下如何使用请求取消,有两种使用方式:

  • 方式一 使用axios.CancelToken的source方法,调用后返回token和cancel两个属性,token用于请求时传给配置对象中的cancelToken属性,在请求发出后,可以使用cancel方法取消请求。
const CancelToken = axios.CancelToken;
const source = CancelToken.source();
axios.get('/test/get', {
  cancelToken: source.token
})
source.cancel('Operation canceled by the user.');
  • 方式二 直接new一个CancelToken的实例赋给配置对象的cancelToken属性并传入一个函数,该函数接收一个处理取消逻辑的函数参数,在函数体内将其赋给在外手动声明的cancel变量,通过执行cancel函数取消请求。
const CancelToken = axios.CancelToken;
let cancel;
axios.get('/test/get', {
  cancelToken: new CancelToken(function executor(c) {
    cancel = c;
  })
});
cancel();

接下来分析其实现思路。
我们知道http请求取消是通过调用xhr对象的abort方法实现的。现在的问题是当想要取消请求时我们往往无法直接访问xhr对象。要想操作xhr对象我们只有在请求发送的过程中,也即在new XMLHttpRequest()后才能访问xhr对象。而要实现请求取消,我们只能事先将请求取消的逻辑写好但不执行,等将来需要取消时再执行这段逻辑。说到这里就又想到了promise,我们知道promise.then方法中指定的回调是会等promise的状态改变后再执行的,因此实现思路也就有啦。我们可以声明一个pending状态的promise,在then方法的成功回调中添加请求取消的逻辑,即xhr.abort。等需要取消时,将该promise的状态改变即可。换句话说我们把请求取消的逻辑"寄托"在了一个promise上。那么这个promise从何而来?观察拦截器的两种使用方式,其实配置对象中的cancelToken属性就是那个promise。

既然有两种使用方式,相应的也就有两种方式可以得到这个promise,分别是直接实例化axios.CancelToken类以及调用CancelToken类的source方法得到。axios得到这个promise后就可以将取消逻辑‘寄托’在其then方法上,接下来看一下实现:

export default (config: AxiosRequestConfig): AxiosPromise => {
  return new Promise((resolve,reject)=>{
    const {
   ......
   cancelToken,// 传入的promise
    } = config
  const request = new XMLHttpRequest()
  // ...其他逻辑
  if (cancelToken) {
        cancelToken.promise.then(reason => {
          // 调用取消方法
          request.abort()
          // 改变Axiospromise状态为失败
          reject(reason)
        })
      }
  })
}

接下里分析如何改变该promise的状态。观察第二种使用方式,executor函数接收的参数就是用来处理取消逻辑的函数,即改变promise的状态。因此在实现CancelToken类时首先需要声明一个pending状态的promise,即暂不执行resolve函数,而是将其暂存起来(赋给外部变量)。之后执行executor函数并传入一个函数,函数内部执行暂存的resolve函数即可改变上述promise的状态,接下来具体实现

interface ResolvePromise {
  (reason?: string): void
}
class CancelToken {
  promise: Promise<string>
  reason?: string
  constructor(executor) {
    let resolvePromise: ResolvePromise
    // 声明一个pending状态的promise
    this.promise = new Promise<string>(resolve => {
      resolvePromise = resolve
    })
    // executor函数传入的参数会被赋值给外部变量cancel用于取消请求
    executor(message => {
      if (this.reason) {
        return
      }
      this.reason = message
      resolvePromise(this.reason)
    })
  }
}

至此第二种使用方式已经实现,再来看第一种。容易看出source函数返回的token就是一个pending状态的promise,返回的cancel函数可直接调用,无需我们手动声明,对比一下不难发现其实这就是相对于第二种方式做了一层封装,将CancelToken的实例化和取消函数的处理逻辑放在source方法内部实现,接下来实现source方法。

class CancelToken {
 ...
  //静态方法source 实例化CancelToken 声明cancel变量 接收取消函数并返回
  static source(): CancelTokenSource {
    let cancel
    const token = new CancelToken(c => {
      cancel = c
    })
    return {
      cancel,
      token
    }
  }
  constructor(executor) {...} 
}

至此请求取消的主体逻辑已经实现。有一点需要特别区分,即axios.CancelToken和config的cancelToken这两个属性,事实上他们是类与实例的关系。

源码地址,如有错误恳请指正。

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

推荐阅读更多精彩内容