对大多数Java开发者来说,Spring事务管理是Spring应用中最常用的功能,使用也比较简单。本文主要逐步介绍Spring事务管理的相关知识点及原理,作为Spring事务管理的学习总结。

一、关键类




Spring AOP和事务管理 spring aop事务原理_Spring AOP和事务管理


事务真正的开始、提交、回滚都是通过PlatformTransactionManager这个接口来实现的,例如,我们常用的org.springframework.jdbc.datasource.DataSourceTransactionManager。

TransactionDefinition用于获取事务的一些属性,Isolation, Propagation,Timeout,Read-only,还定义了事务隔离级别,传播属性等常量。

TransactionStatus用于设置和查询事务的状态,如是否是新事务,是否有保存点,设置和查询RollbackOnly等。

二、声明式事务

所谓声明式事务,就是通过配置的方式省去很多代码,从而让Spring来帮你管理事务。本质上就是配置一个Around方式的AOP,在执行方法之前,用TransactionInterceptor截取,然后调用PlatformTransactionManager的某个实现做一些事务开始前的事情,然后在方法执行后,调用PlatformTransactionManager的某个实现做commit或rollback. 如图:


Spring AOP和事务管理 spring aop事务原理_AOP_02


声明式事务可以通过XML配置,也可以通过Annotation的方式来配置,还可以两种结合。平时项目中看到比较多的是两种结合的方式,在XML中配置数据源,事务管理器,然后AOP相关的通过@Transactional(该注解可以注在Class,Method上)来配置。(个人感觉,AOP相关的配置用XML配置挺繁琐的,还是注解好)例如:


Spring AOP和事务管理 spring aop事务原理_Spring AOP和事务管理_03


三、事务属性

引用官方文档的表格


Spring AOP和事务管理 spring aop事务原理_springaop事务逻辑原理_04


(一)、value,在有多个事务管理器存在的情况下,用于标识使用哪个事务管理器

(二)、isolation,事务的隔离级别,默认是Isolation.DEFAULT,这个DEFAULT是和具体使用的数据库相关的。关于隔离级别,可以参考MySQL事务学习总结

(三)、readOnly, 是否只读,如果配置了true,但是方法里使用了update,insert语句,会报错。对于只读的事务,配置为true有助于提高性能。

(四)、rollbackFor, noRollbackFor. Spring的声明式事务的默认行为是如果方法抛出RuntimeException或者Error,则事务会回滚,对于其他的checked类型的异常,不会回滚。如果想改变这种默认行为,可以通过这几个属性来配置。

(五)、propagation, 后面会具体讲。

四、 事务的传播机制


Spring AOP和事务管理 spring aop事务原理_AOP_05


Spring AOP和事务管理 spring aop事务原理_为什么spring事务管理不回溯_06


其他的都还好理解,后面结合例子重点介绍下PROPAGATION_REQUIRED,PROPAGATION_REQUIRES_NEW,PROPAGATION_NESTED三种传播级别。

表结构和原始数据:


Spring AOP和事务管理 spring aop事务原理_Spring AOP和事务管理_07


(一)、PROPAGATION_REQUIRED


Spring AOP和事务管理 spring aop事务原理_Spring AOP和事务管理_08


执行完之后,test表的数据没有任何变化。

由于MysqlTest02中的事务传播类型是Propagation.REQUIRED,逻辑上有两个事务,但底层是共用一个物理事务的,第二个事务的抛出RuntimeExcetion导致事务回滚,对于这种传播类型,内层事务的回滚会导致外层事务回滚。所以数据库中的数据没有任何变化。

(二)、PROPAGATION_REQUIRES_NEW


Spring AOP和事务管理 spring aop事务原理_springaop事务逻辑原理_09


同样的代码,唯一的区别就是第二个事务的传播属性改成了REQUIRES_NEW,执行结果是啥?不好意思,第二个事务执行不了。

对于REQUIRES_NEW,逻辑上有两个事务,底层物理上也有两个事务,由于第一个事务和第二个事务更新的是同一条记录,对于Mysql默认的隔离级别REPEATABLE-READ来说,第一个事务会对该记录加排他锁,所以第二个事务就一直卡住了。

OK,我们把第二个事务的执行的SQL语句换成。


Spring AOP和事务管理 spring aop事务原理_Spring AOP和事务管理_10


执行结果如下,可以看到只有第二个事务回滚了。


Spring AOP和事务管理 spring aop事务原理_AOP_11


(三)、PROPAGATION_NESTED

对于这种传播类型,物理上只有一个事务,不过可以有多个savePoint用来回滚。当然是用这种传播类型,需要数据库支持savePoint,使用jdbc的也是要3.0版本以上(这个不太确定)。


Spring AOP和事务管理 spring aop事务原理_AOP_12


执行结果是如下,可以看到第一个事务和第三个事务提交成功了,第二个事务回滚了。物理上它们是在一个事务里的,只不过用到了保存点的技术。


Spring AOP和事务管理 spring aop事务原理_AOP_13


五、其他

在写测试代码的时候遇到了一个关于AOP的问题,可以看到我的测试代码,每个事务都是在一个新的class中写的。为什么不像下面这样写呢?


Spring AOP和事务管理 spring aop事务原理_事务管理_14


这是因为在Spring的AOP中,test01调用test02, test02是不会被AOP截获的,所以也不会被Spring进行事务管理。原因是Spring AOP的实现本质是通过动态代理的方式去执行真正的方法,然后在代理类里面做一些额外的事情。当通过别的类调用MysqlTest01中的test01方法时,因为使用了Spring的DI,注入的其实是一个MysqlTest01的一个代理类,而通过内部方法调用test02时,则不是。

6. Reference


Spring AOP和事务管理 spring aop事务原理_springaop事务逻辑原理_15


Spring Framework Reference Documentation