天天看点

死磕Spring之AOP篇 - Spring AOP总览

该系列文章是本人在学习 Spring 的过程中总结下来的,里面涉及到相关源码,可能对读者不太友好,请结合我的源码注释 Spring 源码分析 GitHub 地址 进行阅读。

Spring 版本:5.1.14.RELEASE

在开始阅读 Spring AOP 源码之前,需要对 Spring IoC 有一定的了解,可查看我的 《死磕Spring之IoC篇 - 文章导读》 这一系列文章

了解 AOP 相关术语,可先查看 《Spring AOP 常见面试题) 》 这篇文章

该系列其他文章请查看:《死磕 Spring 之 AOP 篇 - 文章导读》

通过上一篇 《初识 JDK、CGLIB 两种动态代理》 文章我们对 Spring AOP 底层的 JDK 动态代理和 CGLIB 动态代理有了一定的了解,也知道如何简单地使用两种动态代理创建代理对象。相信上篇文章可以让你对 Spring AOP 有了一个初步的认识,那么接下来我们准备进入 Spring AOP 源码学习阶段。

在开始 Spring AOP 源码学习前,本文会对 Spring AOP 涉及到的大部分主要的 API 进行介绍,让我们对 Spring AOP 有一个全面的认识,这样在学习 Spring AOP 源码的过程中有一个清晰的思路会更加易于理解,然后在后续的文章从 Spring AOP 的自动代理开始深入学习,直接进入正题吧。

Spring AOP API 总览

如下图所示:

死磕Spring之AOP篇 - Spring AOP总览

上面这张图片列出了 Spring AOP 涉及到的大部分 API,接下来我们依次简单介绍一下:

  • Joinpoint:连接点,也就是我们需要执行的目标方法
  • Pointcut:切点,提供 ClassFilter 类过滤器和 MethodMatcher 方法匹配器支持对类和方法进行筛选。
  • Advice:通知器,一般都是 MethodInterceptor 方法拦截器,不是的话会通过 AdvisorAdapter 适配器转换成对应的 MethodInterceptor 方法拦截器。
  • Advisor:Advice 容器接口,与 Advice 是一对一的关系;它的子接口 PointcutAdvisor 相当于一种 Pointcut 和 Advice 的容器,将 Pointcut 过滤 Joinpoint 的能力和 Advice 进行整合,这样一来就将两者关联起来了。
  • Advisor 适配器:AdvisorAdapter 是 Advisor 的适配器,当筛选出能够应用于方法的所有 Advisor 后,需要获取对应的 Advice;
    • 如果不是 MethodInterceptor 类型,则需要通过 AdvisorAdapter 适配器转换成对应的 MethodInterceptor 方法拦截器。
  • AOP 代理对象:AOP 代理对象,由 JDK 动态代理或者 CGLIB 动态代理创建的代理对象;
    • 选择哪种动态代理是通过 AopProxyFactory 代理工厂根据目标类来决定的。
  • AOP 代理配置:AdvisedSupport 配置管理器中保存了对应代理对象的配置信息,例如满足条件的 Advisor 对象、TargetSource 目标类来源;
    • 在创建代理对象的过程,AdvisedSupport 扮演着一个非常重要的角色。
  • AOP 代理对象创建:AOP 代理对象的创建分为手动模式和自动模式;不管哪种模式都和 AdvisedSupport 配置管理器有关;
    • 手动模式就是通过 Spring AOP 提供的 API 进行创建;
    • 自动模式则是和 Spring IoC 进行整合,在 Bean 的加载过程中如果需要进行代理,则创建对应的代理对象;
  • AdvisorChainFactory:Advisor 链路工厂
    • AdvisedSupport 配置管理器中保存了代理对象的所有 Advisor 对象,当拦截某个方法时,需要通过 AdvisorChainFactory 筛选出能够应用于该方法的 Advisor 们;
    • 另外还需要借助 AdvisorAdapter 适配器获取 Advisor 对应的 MethodInterceptor 方法拦截器,将这些 MethodInterceptor 有序地形成一条链路并返回。
  • IntroductionInfo:引介接口,支持代理对象实现目标类没有真正实现的额外的接口;
    • 在 Advisor 的子接口 IntroductionAdvisor 中会继承这个 IntroductionInfo 接口,通过 @DeclareParents 注解定义的字段会被解析成 IntroductionAdvisor 对象。
  • AOP 代理目标对象来源:目标类来源,和代理对象进行关联,用于获取被代理代理的目标对象;
    • 在代理对象中最终的方法执行都需要先通过 TargetSource 获取对应的目标对象,然后执行目标方法。

Joinpoint

类图

死磕Spring之AOP篇 - Spring AOP总览
  • ConstructorInvocation,是基于构造器的一个调用器,在 Spring AOP 中仅支持方法级别的代理,所以这个接口并没有被使用
  • ReflectiveMethodInvocation,Joinpoint 的底层实现类,一个基于反射的方法调用器,本文不进行分析,在后续的文章进行讲解