事务传播行为 (Propagation Behavior)
事务传播行为指的是当一个事务方法被另一个事务方法调用时,应该如何处理事务。常见场景是 Service A 里面的一个事务方法调用 Service B 里面的一个事务方法。根据不同需求,可以有几种解决方案:
- 融入事务:直接去掉 Service B 中关于开启事务和提交事务的
begin
和commit
,融入到 Service A 的事务中。问题:B 事务的错误会引起 A 事务的回滚。 - 挂起事务:如果不想 B 事务的错误引起 A 事务的回滚,可以开启两个连接,一个执行 A,一个执行 B,互不影响。执行到 B 的时候把 A 挂起,新起连接去执行 B,B 执行完再唤醒 A 执行。
- 嵌套事务:在 MySQL 中通过给 B 事务加
savepoint
和rollback
来模拟嵌套事务,把 B 设置成伪事务。
Spring 中的事务传播行为
Spring 提供了 7 种事务传播行为:
1. PROPAGATION_REQUIRED
(需要)
- 需要什么?:需要事务。
- 行为:如果存在一个事务,则直接使用当前事务;如果没有事务,则开启一个新的事务。
- 应用场景:
- 如果外层方法有事务,内层方法会融入外层事务。
- 如果外层没有事务,内层会新建一个事务。
2. PROPAGATION_SUPPORTS
(支持)
- 支持什么?:支持外层事务。
- 行为:如果存在一个事务,支持当前事务;如果没有事务,则非事务执行。
- 应用场景:
- 如果外层方法有事务,内层方法融入外层事务。
- 如果外层没有事务,内层方法会以非事务方式执行。
3. PROPAGATION_MANDATORY
(强制性)
- 强制什么?:强制必须有事务。
- 行为:如果当前存在一个事务,则使用当前事务;如果没有活动的事务,则抛出异常。
- 应用场景:
- 如果外层方法有事务,内层方法融入外层事务。
- 如果外层没有事务,内层方法会抛出
IllegalTransactionStateException
异常。