• Spring-Spring之AOP底层源码解析(下)


    ProxyFactory选择cglib或jdk动态代理原理

    ProxyFactory在生成代理对象之前需要决定到底是使用JDK动态代理还是CGLIB技术:

    1. // config就是ProxyFactory对象
    2. // optimize为true,或proxyTargetClass为true,或用户没有给ProxyFactory对象添加interface
    3. if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
    4.  Class<?> targetClass = config.getTargetClass();
    5.  if (targetClass == null) {
    6.   throw new AopConfigException("TargetSource cannot determine target class: " +
    7.     "Either an interface or a target is required for proxy creation.");
    8.  }
    9.     // targetClass是接口,直接使用Jdk动态代理
    10.  if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) {
    11.   return new JdkDynamicAopProxy(config);
    12.  }
    13.     // 使用Cglib
    14.  return new ObjenesisCglibAopProxy(config);
    15. }
    16. else {
    17.     // 使用Jdk动态代理
    18.  return new JdkDynamicAopProxy(config);
    19. }

    代理对象创建过程

    JdkDynamicAopProxy

    1. 在构造JdkDynamicAopProxy对象时,会先拿到被代理对象自己所实现的接口,并且额外的增加SpringProxy、Advised、DecoratingProxy三个接口,组合成一个Class[],并赋值给proxiedInterfaces属性
    2. 并且检查这些接口中是否定义了equals()、hashcode()方法
    3. 执行Proxy.newProxyInstance(classLoader, this.proxiedInterfaces, this),得到代理对象,JdkDynamicAopProxy作为InvocationHandler,代理对象在执行某个方法时,会进入到JdkDynamicAopProxy的**invoke()**方法中

    ObjenesisCglibAopProxy

    1. 创建Enhancer对象
    2. 设置Enhancer的superClass为通过ProxyFactory.setTarget()所设置的对象的类
    3. 设置Enhancer的interfaces为通过ProxyFactory.addInterface()所添加的接口,以及SpringProxy、Advised、DecoratingProxy接口
    4. 设置Enhancer的Callbacks为DynamicAdvisedInterceptor
    5. 最后创建一个代理对象,代理对象在执行某个方法时,会进入到DynamicAdvisedInterceptor的intercept()方法中

    代理对象执行过程

    1. 在使用ProxyFactory创建代理对象之前,需要往ProxyFactory先添加Advisor
    2. 代理对象在执行某个方法时,会把ProxyFactory中的Advisor拿出来和当前正在执行的方法进行匹配筛选
    3. 把和方法所匹配的Advisor适配成MethodInterceptor
    4. 把和当前方法匹配的MethodInterceptor链,以及被代理对象、代理对象、代理类、当前Method对象、方法参数封装为MethodInvocation对象
    5. 调用MethodInvocation的proceed()方法,开始执行各个MethodInterceptor以及被代理对象的对应方法
    6. 按顺序调用每个MethodInterceptor的invoke()方法,并且会把MethodInvocation对象传入invoke()方法
    7. 直到执行完最后一个MethodInterceptor了,就会调用invokeJoinpoint()方法,从而执行被代理对象的当前方法

    各注解对应的MethodInterceptor

    • @Before对应的是AspectJMethodBeforeAdvice,在进行动态代理时会把AspectJMethodBeforeAdvice转成MethodBeforeAdviceInterceptor
      • 先执行advice对应的方法
      • 再执行MethodInvocation的proceed(),会执行下一个Interceptor,如果没有下一个Interceptor了,会执行target对应的方法
    • @After对应的是AspectJAfterAdvice,直接实现了MethodInterceptor
      • 先执行MethodInvocation的proceed(),会执行下一个Interceptor,如果没有下一个Interceptor了,会执行target对应的方法
      • 再执行advice对应的方法
    • @Around对应的是AspectJAroundAdvice,直接实现了MethodInterceptor
      • 直接执行advice对应的方法,由@Around自己决定要不要继续往后面调用
    • @AfterThrowing对应的是AspectJAfterThrowingAdvice,直接实现了MethodInterceptor
      • 先执行MethodInvocation的proceed(),会执行下一个Interceptor,如果没有下一个Interceptor了,会执行target对应的方法
      • 如果上面抛了Throwable,那么则会执行advice对应的方法
    • @AfterReturning对应的是AspectJAfterReturningAdvice,在进行动态代理时会把AspectJAfterReturningAdvice转成AfterReturningAdviceInterceptor
      • 先执行MethodInvocation的proceed(),会执行下一个Interceptor,如果没有下一个Interceptor了,会执行target对应的方法
      • 执行上面的方法后得到最终的方法的返回值
      • 再执行Advice对应的方法

    AbstractAdvisorAutoProxyCreator

    DefaultAdvisorAutoProxyCreator的父类是AbstractAdvisorAutoProxyCreator。

    AbstractAdvisorAutoProxyCreator非常强大以及重要,只要Spring容器中存在这个类型的Bean,就相当于开启了AOP,AbstractAdvisorAutoProxyCreator实际上就是一个BeanPostProcessor,所以在创建某个Bean时,就会进入到它对应的生命周期方法中,比如:在某个Bean初始化之后,会调用wrapIfNecessary()方法进行AOP,底层逻辑是,AbstractAdvisorAutoProxyCreator会找到所有的Advisor,然后判断当前这个Bean是否存在某个Advisor与之匹配(根据Pointcut),如果匹配就表示当前这个Bean有对应的切面逻辑,需要进行AOP,需要产生一个代理对象。

    @EnableAspectJAutoProxy

    这个注解主要就是往Spring容器中添加了一个AnnotationAwareAspectJAutoProxyCreator类型的Bean。
     

    AnnotationAwareAspectJAutoProxyCreator.png


    AspectJAwareAdvisorAutoProxyCreator继承了AbstractAdvisorAutoProxyCreator,重写了findCandidateAdvisors()方法,AbstractAdvisorAutoProxyCreator只能找到所有Advisor类型的Bean对象,但是AspectJAwareAdvisorAutoProxyCreator除开可以找到所有Advisor类型的Bean对象,还能把@Aspect注解所标注的Bean中的@Before等注解及方法进行解析,并生成对应的Advisor对象。

    所以,我们可以理解@EnableAspectJAutoProxy,其实就是像Spring容器中添加了一个AbstractAdvisorAutoProxyCreator类型的Bean,从而开启了AOP,并且还会解析@Before等注解生成Advisor。

    Spring中AOP原理流程图

    Spring中AOP底层原理| ProcessOn免费在线作图,在线流程图,在线思维导图

  • 相关阅读:
    CorelDRAW2023安装下载教程精简版矢量绘图软件
    某安全设备frp流量告警分析
    [创业-40]:-优秀人与普通人的区别
    Kafka磁盘写满日志清理操作
    文举论金:黄金原油全面走势分析策略指导。
    [蓝桥杯 2022 省 A] 求和
    Java常见跳出循环的4种方式总结、switch中的break与return、lamada表达式中foreach如何正确选择退出方式
    list的用法
    Docker实战-第一章欢迎来到Docker世界
    LeetCode讲解篇之面试题 10.11. 峰与谷
  • 原文地址:https://blog.csdn.net/weixin_43874650/article/details/134431807