当前位置: 首页 > news >正文

【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下的文件中,定义了一长串自动配置类,其中就包括:

org.springframework.boot.autoconfigure.aop.AopAutoConfiguration

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

  • 这个类上有一个条件注解 @ConditionalOnClass(Advice.class),由于Starter引入了AOP包,这个条件百分百满足。
  • AopAutoConfiguration 内部使用了一个关键注解 @EnableAspectJAutoProxy。这个注解的作用,就是向Spring容器注册 AnnotationAwareAspectJAutoProxyCreator 的Bean定义。
@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下的文件中,定义了一长串自动配置类,其中就包括:

org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration

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

@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 {// 关键在这里!@EnableTransactionManagementpublic static class ProxyTransactionManagementConfiguration {}}
}

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

@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),即我们的TransactionInterceptoradvisor.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)。

http://www.dtcms.com/a/357172.html

相关文章:

  • 球型摄像机实现360°无死角
  • 【前端教程】从基础到专业:诗哩诗哩网HTML视频页面重构解析
  • 技术干货|Prometheus告警及告警规则
  • APM32芯得 EP.31 | APM32F402 HC-SR04超声测距经典操作:波形输出与滤波
  • 微算法科技(NASDAQ:MLGO)一种基于FPGA的Grover搜索优化算法技术引领量子计算
  • PCIe 6.0配置与地址空间架构:深入解析设备初始化的核心机制
  • C#实现OPC客户端
  • 《Password Guessing Using Random Forest》论文解读
  • system论文阅读--HPCA25
  • Excel Word Pdf 格式转换
  • ubuntu 安装 vllm
  • 电平移位器的原理
  • 群核科技--SpatialGen
  • pytest使用allure测试报告
  • Pytest 插件方法:pytest_runtest_makereport
  • 多方调研赋能AI+智慧消防 豪越科技人工智能创新获认可
  • 【网络安全领域】边界安全是什么?目前的发展及应用场景
  • java基本类型关键字
  • EasyExcel处理大数据量导出
  • 新手法务合同审查,有什么建议?
  • 单点登录(SSO)前端(Vue2.X)改造
  • 关于锁相放大器(LIA)的系统论文研究(重点于FPGA部分)
  • 设计模式:装饰模式(Decorator Pattern)
  • iOS开发之苹果系统包含的所有字体库
  • 最小生成树——Kruskal
  • 【机器学习入门】3.1 关联分析——从“购物篮”到推荐系统的核心逻辑
  • 响应式编程框架Reactor【2】
  • Windows C盘完全占满会如何?
  • 2024-06-13-debian12安装Mariadb-Galera-Cluster+Nginx+Keepalived高可用多主集群
  • 毕马威 —— 公众对人工智能的信任、态度及使用情况调查