事务的七种传播方式
所谓事务的传播行为是指,如果在开始当前事务之前,一个事务上下文已经存在,此时有若干选项可以指定一个事务性方法的执行行为。
当事务存在嵌套关系时,子事务与父事务的关系以及回滚的影响范围(传播-回滚的传播)
示例定义:
T1{
O(A);
T2{
O(B);
O(C);
};
O(D);
};
各种传播性的关系
PROPAGATION_REQUIRED=0:
如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。
- 无父事务时:子事务作为独立事务执行
- 有父事务时:子事务中的操作串入父事务中执行,并且一起提交,一个操作失败全部回滚
总结:如果有父事务就和父事务同进退,如果没有就新建事务
PROPAGATION_SUPPORTS=1:
如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
- 无父事务时:以非事务方式执行
- 有父事务时:加入父事务执行,等同于
PROPAGATION_REQUIRED
总结:如果有父事务就和父事务同进退,如果没有就按非事务执行
PROPAGATION_MANDATORY=2:
如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。
- 无父事务时:抛出异常
- 有父事务时:加入父事务执行,等同于
PROPAGATION_REQUIRED
总结:如果有父事务就和父事务同进退,如果没有就抛出异常
PROPAGATION_REQUIRES_NEW=3:
创建一个新的事务,如果当前存在事务,则把当前事务挂起。
挂起(Suspend):通知TransactionManager
不再检查某事务的状态,直到Resume
AbstractPlatformTransactionManager. handleExistingTransaction()
- 无父事务时:子事务新建事务作为独立事务执行
- 有父事务时:子事务新建事务作为独立事务执行,独立提交;
T1{
O(A);
T2{
O(B);
O(C);
};
O(D);
};
如此示例:
- 子事务T2不受父事务T1回滚的影响,但仍作为T1的子逻辑,
- O(D)失败,O(A)回滚,T2中的事务不回滚;
- T2失败回滚,T1捕获异常后,可以选择提交或回滚,未捕获异常,同T2一起回滚。
总结:新建事务不管有没有父事务,互不影响(父事务通过捕获异常可以选择是否回滚)
NOT_SUPPORTED=4:
以非事务方式运行,如果当前存在事务,则把当前事务挂起。
- 无父事务时:以非事务方式执行
- 有父事务时:挂起父事务,自己按照无事务方式运行
子事务自身无回滚,出现异常若向上抛,可能导致父事务回滚
父事务回滚时,不会影响子事务。
总结:不管有没有父事务都按非事务运行,如果出现异常,可能导致父事务回滚
NEVER=5:
以非事务方式运行,如果当前存在事务,则抛出异常。
- 无父事务时:以非事务方式执行
- 有父事务时:抛出异常(若不处理会导致父事务回滚)
总结:按非事务运行,如果有父事务存在就抛出异常,如果父事务不处理就导致父事务回滚
NESTED=6:
如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行
- 无父事务时:创建独立事务,等同于PROPAGATION_REQUIRED
- 有父事务时:嵌套在父事务之内
子事务依赖父事务:子事务于父事务提交时提交;父事务回滚,子事务也回滚。
Savepoint:子事务回滚时,父事务不回滚
总结:听爹的,爹干啥儿干啥,但是爹不听儿的