Spring中申明式事务(@Transactional)实现的原理解析

前言

在Springboot中使用事务可以通过配置的方式以及声明式事务的方式去开启一个事务,对于申明式事务的使用,只需要在方法上面添加一个**@Transactional** 注解即可开启事务。但是让你看到最简单的事务,其实最神秘,如果没有真正的了解它也很容易使用错误导致出现问题。例如:什么情况下会导致事务失效、@Transactional注解中的各个属性应该在什么情况下使用(事务的传播特性、readOnly等)。只有真正的了解了声明式事务的原理之后才能够放心的去使用它。


一、Springboot中事务是如何自动配置

pringboot中的自动化配置都是通过在spring-boot-autoconfigure中的spring.factories文件中通过spi的方式进行引入的。打开这个spring.factories文件可以看到里面有一个TransactionAutoConfiguration,直接查看这个类的源码。

这里面引入了一个**@EnableTransactionManagement** 注解,之前使用Spring的时候就需要添加**@EnableTransactionManagement**去开启事务,所以这个注解是开启事务的入口。

这个注解通过@Import引入了TransactionManagementConfigurationSelector

这个类继承了AdviceModeImportSelector ,而AdviceModeImportSelector 是实现了ImportSelector 接口的。所以这个selectImports方法是想spring容器中注入了AutoProxyRegistrarProxyTransactionManagementConfiguration。

二、 AutoProxyRegistrar的工作原理

AutoProxyRegistrar 实现了ImportBeanDefinitionRegistrar 接口,在Spring的扩展类中,ImportBeanDefinitionRegistrar 接口是实现了想Spring容器中注入Bean的作用,而这里向Spring容器中注入了AnnotationAwareAspectJAutoProxyCreatorInfrastructureAdvisorAutoProxyCreator。AbstractAdvisorAutoProxyCreator 这个类就是创建Aop增强的关键。在之前关于Spring Aop的文章 《SpringBoot的AOP原理》中就已经说过。事务其实也是一种AOP增强,但是事务增强的实现是通过InfrastructureAdvisorAutoProxyCreator类来创建增强的。

根据 《SpringBoot的AOP原理》这篇文章,可以知道,Springboot是通过扩展点去一步一步的自行AOP增强的相关方法,先通过AbstractAutoProxyCreator#postProcessBeforeInstantiation 方法中获取切点和切面。这里的逻辑就跟AOP一模一样,都是调用的父类AbstractAutoProxyCreator 中的方法逻辑。唯一的不同点在于InfrastructureAdvisorAutoProxyCreator 没有重写findCandidateAdvisors方法,这个方法的主要逻辑是寻找Advisor的对象。

三、事务增强的代理类创建逻辑

前面已经完成了接入点和切面的查找了,后面就是怎么将这些切面织入到切入点,就是 《SpringBoot的AOP原理》中所说的创建代理对象的过程。这一步的主要逻辑就在getAdvicesAndAdvisorsForBean以及createProxy方法中,具体可以看一位大神写的文章《Spring源码深度解析》。而事务的具体逻辑就在Advisor和pointcut中,这一部分下次再说。


总结

事务其实也是AOP增强的一种,所以实现的原理也是跟AOP一样,只是具体的切面以及切入点的不同。

相关推荐
CUIYD_19893 分钟前
Eclipse 常用搜索功能汇总
java·ide·eclipse
我不只是切图仔1 小时前
我只是想给网站加个注册验证码,咋就那么难!
前端·后端
专注VB编程开发20年1 小时前
CSS 的命名方式像是 PowerShell 的动词-名词结构,缺乏面向对象的层级关系
开发语言·后端·rust
野犬寒鸦1 小时前
力扣hot100:相交链表与反转链表详细思路讲解(160,206)
java·数据结构·后端·算法·leetcode
爱吃烤鸡翅的酸菜鱼1 小时前
【Spring】原理:Bean的作用域与生命周期
后端·spring
JohnYan1 小时前
工作笔记 - 微信消息发送和处理
javascript·后端·微信
该用户已不存在1 小时前
macOS是开发的终极进化版吗?
前端·后端
ytadpole2 小时前
揭秘设计模式:工厂模式的五级进化之路
java·设计模式
计算机毕业设计木哥2 小时前
计算机毕设选题:基于Python+Django的B站数据分析系统的设计与实现【源码+文档+调试】
java·开发语言·后端·python·spark·django·课程设计
失散132 小时前
分布式专题——1.2 Redis7核心数据结构
java·数据结构·redis·分布式·架构