@Transactional
1、Spring中对于事务传播性的定义
- PROPAGATION_REQUIRED:支持当前事务,如果当前没有事务,则新建一个事务
- PROPAGATION_SUPPORTS:支持当前事务,如果当前没有事务,则以非事务进行
- PROPAGATION_MANDATORY:支持当前事务,如果当前没有事务,则抛异常
- PROPAGATION_REQUIRES_NEW:新建事务,如果当前存在事务,则把当前事务挂起
- PROPAGATION_NOT_SUPPORTED:以非事务进行,如果当前存在事务,则挂起
- PROPAGATION_NEVER:以非事务进行,如果当前存在事务,则抛异常
- PROPAGATION_NESTED:如果当前存在事务,则在嵌套事务内执行。如果没有,则进行与PROPAGATION_REQUIRED类似操作。
2、PROPAGATION_REQUIRES_NEW 与 PROPAGATION_NESTED区别
PROPAGATION_REQUIRES_NEW 启动一个新的,不依赖环境的内部事务,这个事务将会被完全commit或者rollback而不依赖外部事务。它拥有自己的隔离范围,自己的锁,当内部事务开始执行时,外部事务将被挂起,内部事务结束时,外部事务将继续。
PROPAGATION_NESTED开始一个嵌套的事务,它是已经存在的一个事务的子事务,嵌套事务开始执行时,它将取得一个savepoint,如果这个嵌套事务失败,内部事务将回滚到此savepoint,嵌套事务是外部事务的一部分,只有外部事务结束后它才会被提交。
例:
A方法调用B方法
异常状态 | PROPAGATION_REQUIRES_NEW | PROPAGATION_NESTED |
---|---|---|
A异常,B正常 | A回滚,B正常提交 | A与B一起回滚 |
A正常,B异常 | (1)如果A捕获B的异常,则B先回滚,A再正常提交 (2)如果A未捕获,则B先回滚,A再回滚 | B先回滚,A再正常提交 |
A异常,B异常 | B先回滚,A再回滚 | A与B一起回滚 |
A正常,B正常 | B先提交,A再提交 | A与B一起提交 |
3、@Transactional主方法调用子方法事务的有无
3.1 两个方法在同一个类中
- 主有子有 => 有 全部回滚
- 主有子无 => 有 全部回滚
- 主无子有 => 无
- 主无子无 => 无
以上说明,同一个类中子方法@Transactional都是不生效的,取决于主方法有没有事务,哪怕加上了参数(propagation= Propagation.PROPAGATION_REQUIRES_NEW )也是没用的。
3.2 两个方法在不同类中
- 主有子有 => 有 全部回滚
- 主有子无 => 有 全部回滚
- 主无子有 => 子有
- 主无子无 => 无
此时,若在子方法上加上(propagation= Propagation.PROPAGATION_REQUIRES_NEW ),则子方法会创建一个新的事务,抛出异常时,只有子方法会回滚。