【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)。

相关推荐
他日若遂凌云志5 分钟前
深入拆解 Windows Socket 五种 I/O 模型:核心机制、Linux 差异与场景适配
后端
小码编匠6 分钟前
开箱即用!集成 YOLO+OpenCV+OCR 的 WebAI 平台(支持RTSP/RTMP视频流识别与自训练)
spring boot·后端·opencv
zcychong6 分钟前
如何让A、B、C三个线程按严格顺序执行(附十一种解)?
java·面试
蜚鸣9 分钟前
Spring依赖注入方式
spring
文心快码BaiduComate26 分钟前
再获殊荣!文心快码荣膺2025年度优秀软件产品!
前端·后端·代码规范
步行cgn31 分钟前
HttpSessionBindingListener
java·开发语言·数据仓库·servlet
浮游本尊37 分钟前
Java学习第24天 - Spring Cloud Gateway与容器化部署
java
天天摸鱼的java工程师1 小时前
SpringBoot + RabbitMQ + Redis + MySQL:社交平台私信发送、已读状态同步与历史消息缓存
java·后端
Kiri霧1 小时前
Rust数组与向量
开发语言·后端·rust
特立独行的猫a1 小时前
Rust语言入门难,难在哪?所有权、借用检查器、生命周期和泛型介绍
开发语言·后端·rust