【Spring底层分析】Spring AOP补充以及@Transactional注解的底层原理分析

这篇文章主要是针对源码进行总结,如果想看源码可以移步到另外一篇文章:https://blog.csdn.net/m0_73866527/article/details/148384920?spm=1001.2014.3001.5501

一、AOP底层原理补充

1、@EnableAspectJAutoProxy,依赖springboot自动装配原理。

项目中引入了 Spring Boot 的 依赖比如spring-boot-starter-web,这个依赖间接引入了 spring-boot-autoconfigure 这个JAR包,这个JAR包里的 META-INF下的文件中,定义了一长串自动配置类,其中就包括:

text 复制代码
org.springframework.boot.autoconfigure.aop.AopAutoConfiguration

当springboot启动后,会扫描这个文件里定义的每个类,当处理到AopAutoConfiguration这个类时,做了以下事:

  • 这个类上有一个条件注解 @ConditionalOnClass(Advice.class),由于Starter引入了AOP包,这个条件百分百满足。
  • AopAutoConfiguration 内部使用了一个关键注解 @EnableAspectJAutoProxy 。这个注解的作用,就是向Spring容器注册 AnnotationAwareAspectJAutoProxyCreator 的Bean定义。
java 复制代码
@AutoConfiguration
@ConditionalOnProperty(prefix = "spring.aop", name = "auto", havingValue = "true", matchIfMissing = true)
public class AopAutoConfiguration {

    @Configuration(proxyBeanMethods = false)
    @ConditionalOnClass(Advice.class)
    static class AspectJAutoProxyingConfiguration {

        @Configuration(proxyBeanMethods = false)
        @EnableAspectJAutoProxy(proxyTargetClass = false) // 看这里!
        @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "false")
        static class JdkDynamicAutoProxyConfiguration {
        }

        @Configuration(proxyBeanMethods = false)
        @EnableAspectJAutoProxy(proxyTargetClass = true) // 看这里!
        @ConditionalOnProperty(prefix = "spring.aop", name = "proxy-target-class", havingValue = "true", matchIfMissing = true)
        static class CglibAutoProxyConfiguration {
        }
    }
    // ... 其他配置
}

总结:@EnableAspectJAutoProxy -> 作用是向spring容器中注册AnnotationAwareAspectJAutoProxyCreator后置处理器的bean定义。

(以上代码可以发现有个熟悉字眼:jdk、cglib,这里挖个坑,我们下次来学习这两种代理方式~)

2、spring容器启动后,流程是:

  • 扫描指定包路径下的注解,然后将被注解标识的类信息保存到bean定义中

  • 注册所有后置处理器到容器中

  • 注册所有非懒加载单实例bean

3、注册所有单实例bean的流程:

  • 实例化
  • 属性赋值
  • 初始化

4、在初始化之前,会遍历所有后置处理器执行postProcessBeforeInstantiation方法。这里我们具体介绍AnnotationAwareAspectJAutoProxyCreator.postProcessBeforeInstantiation:

  • 遍历容器中所有已注册的Bean定义
  • 找出被 @Aspect 注解标注的类
  • 解析这个类上的所有增强方法,将切入点、增强方法、增强类型缓存起来。

5、初始化之后,会遍历所有后置处理器执行postProcessAfterInitialization方法。这里我们具体介绍AnnotationAwareAspectJAutoProxyCreator.postProcessAfterInitialization:

  • 查找基础设施Advisor:比如已经存在于容器中的 TransactionAdvisor

  • 构建 并 返回自定义的AspectJ Advisor:

    • 遍历容器中所有已经实例化好的单例Bean,对于那些被 @Aspect 注解标注的Bean,会根据之前缓存起来的切面类元数据创建出一个个增强器,一个增强器里包含一个增强方法和pointcut切入点表达式的信息。这些增强器会被缓存起来,用于是否创建代理对象的判断。

    至此,我们得到了一个包含"基础设施 Advisor"和"所有自定义 AspectJ Advisor"的完整候选列表。

  • 筛选适用于当前 Bean 的 Advisor:

    • 遍历得到的所有增强器列表
    • 对于每一个增强器,判断这个增强器里的pointcut是否与当前bean匹配。
    • 最后会得到一个筛选后的、只包含适用于当前 Bean 的 Advisor 的列表
    • 对筛选出的增强器列表进行排序
    • 创建代理对象,排序好的增强器列表会与这个代理对象关联起来

6、当调用代理对象的方法时,就会根据这个代理对象相关联的增强器列表构建出拦截器链,然后按顺序递归调用每一个拦截器,就能实现各个增强器的执行时机顺序。

二、@Transactional注解的底层原理

以上过程中,*查找基础设施Advisor:比如已经存在于容器中的 TransactionAdvisor。*对于这一步,你可能有疑惑,这个TransactionAdvisor增强器是啥时候存进容器里的?所以我们接下来分析@Transactional注解的底层原理。

1、@EnableTransactionManagement,依赖springboot自动装配原理。

项目中引入了 Spring Boot 的 依赖比如spring-boot-starter-web,这个依赖间接引入了 spring-boot-autoconfigure 这个JAR包,这个JAR包里的 META-INF下的文件中,定义了一长串自动配置类,其中就包括:

text 复制代码
org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration

当springboot启动后,会扫描这个文件里定义的每个类,当处理到TransactionAutoConfiguration这个类时,做了以下事:

java 复制代码
@AutoConfiguration
@ConditionalOnClass({ PlatformTransactionManager.class })
@EnableConfigurationProperties(TransactionProperties.class)
public class TransactionAutoConfiguration {

    // 这个配置只有在存在PlatformTransactionManager类型的Bean时才会生效
    // 而Spring Boot会自动根据您引入的依赖(如JDBC、JPA)自动创建一个事务管理器Bean

    @Configuration(proxyBeanMethods = false)
    @ConditionalOnBean(PlatformTransactionManager.class)
    @ConditionalOnMissingBean(AbstractTransactionManagementConfiguration.class)
    public static class EnableTransactionManagementConfiguration {

        // 关键在这里!
        @EnableTransactionManagement
        public static class ProxyTransactionManagementConfiguration {
        }
    }
}

ProxyTransactionManagementConfiguration 被解析的过程,实质上是Spring容器执行其内部 @Bean 方法,从而实例化和组装事务核心组件的过程。

java 复制代码
@Configuration
public class ProxyTransactionManagementConfiguration extends AbstractTransactionManagementConfiguration {

    // Bean 1: 事务属性源 - 负责解析@Transactional注解
    @Bean
    @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
    public TransactionAttributeSource transactionAttributeSource() {
        // 创建一个基于注解的事务属性源
        return new AnnotationTransactionAttributeSource();
    }

    // Bean 2: 事务拦截器 - 包含事务管理的核心逻辑(开启、提交、回滚)
    @Bean
    @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
    public TransactionInterceptor transactionInterceptor(TransactionAttributeSource tas) {
        TransactionInterceptor interceptor = new TransactionInterceptor();
        // 注入事务属性源,这样拦截器才知道如何解析注解
        interceptor.setTransactionAttributeSource(tas);
        // 注入事务管理器(来自父类),真正执行事务操作的组件
        if (this.txManager != null) {
            interceptor.setTransactionManager(this.txManager);
        }
        return interceptor;
    }

    // Bean 3: 事务增强器(Advisor) - 连接点(Pointcut)和通知(Advice)的组装体
    @Bean
    @Role(BeanDefinition.ROLE_INFRASTRUCTURE)
    public BeanFactoryTransactionAttributeSourceAdvisor transactionAdvisor(
            TransactionAttributeSource transactionAttributeSource,
            TransactionInterceptor transactionInterceptor) {

        BeanFactoryTransactionAttributeSourceAdvisor advisor = new BeanFactoryTransactionAttributeSourceAdvisor();
        // 关键注入:设置事务属性源,它同时也充当了Pointcut的角色
        advisor.setTransactionAttributeSource(transactionAttributeSource);
        // 关键注入:设置通知(Advice),即我们的TransactionInterceptor
        advisor.setAdvice(transactionInterceptor);
        // 设置Order,决定事务增强在代理链中的位置
        advisor.setOrder(this.enableTx.<Integer>getNumber("order"));
        return advisor;
    }
}
  1. 创建解析器 :首先,执行 transactionAttributeSource() 方法,创建出 AnnotationTransactionAttributeSource。它的职责是解析 @Transactional 注解的属性。

  2. 创建执行器 :接着,执行 transactionInterceptor() 方法。在这个过程中,Spring会将上一步创建的 TransactionAttributeSource 注入给新创建的 TransactionInterceptor。这个拦截器是事务的'执行引擎',包含了事务开启、提交、回滚的核心逻辑。

  3. 创建组装体 :最后,执行 transactionAdvisor() 方法。Spring会将前两步创建好的 TransactionAttributeSourceTransactionInterceptor同时注入给新创建的 BeanFactoryTransactionAttributeSourceAdvisor

    这个Advisor是一个完整的组装体,既拥有了识别哪些方法需要事务 的能力(来自AttributeSource),也拥有了执行事务操作的能力(来自Interceptor)。

相关推荐
bobz9655 分钟前
安装 nvidia 驱动之前要求关闭 secureBoot 么
后端
叫我阿柒啊14 分钟前
Java全栈工程师的面试实战:从技术细节到业务场景
java·数据库·spring boot·微服务·vue·全栈开发·面试技巧
程序员的世界你不懂23 分钟前
【Flask】测试平台开发实战-第一篇
后端·python·flask
CC__xy31 分钟前
《ArkUI 记账本开发:状态管理与数据持久化实现》
java·前端·javascript
布朗克16833 分钟前
OpenTelemetry 通过自动埋点(Java Agent) 应用于springboot项目
java·spring boot·spring·opentelemetry
bobz96536 分钟前
dracut 是什么?
后端
3Cloudream1 小时前
互联网大厂Java面试:从基础到微服务云原生的深度解析
java·spring·微服务·电商·技术架构·面试解析
Eiceblue2 小时前
Java实现PDF表格转换为CSV
java·python·pdf
自由的疯2 小时前
Java RuoYi整合Magic-Api详解
java·后端·架构
自由的疯3 小时前
Java 实现TXT文件上传并解析的Spring Boot应用
后端·架构