前言
Spring在TransactionDefinition接口中规定了7种类型的事务传播行为。事务传播行为是Spring框架独有的事务增强特性,他不属于事务实际提供方数据库的行为。
什么是事务传播行为
事务传播行为: 用来描述由某一个事务传播行为修饰的方法被嵌套进另一个方法的时事务如何传播。
1 |
|
代码中methodA()
方法嵌套调用了methodB()
方法,methodB()
的事务传播行为由@Transaction(Propagation=XXX)
设置决定。这里需要注意的是methodA()
并没有开启事务,某一个事务传播行为修饰的方法并不是必须要在开启事务的外围方法中调用。
Spring中七种事务传播行为
- PROPAGATION_REQUIRED: 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。
Spring默认事务传播
- PROPAGATION_SUPPORTS: 支持当前事务,如果当前没有事务,就以非事务方式执行。
- PROPAGATION_MANDATORY: 使用当前的事务,如果当前没有事务,就抛出异常。
- PROPAGATION_REQUIRES_NEW: 新建事务,如果当前存在事务,把当前事务挂起。
- PROPAGATION_NOT_SUPPORTED: 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
- PROPAGATION_NEVER: 以非事务方式执行,如果当前存在事务,则抛出异常。
- PROPAGATION_NESTED: 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与PROPAGATION_REQUIRED类似的操作。
代码验证
REQUIRED
1 |
|
1 |
|
REQUIRES_NEW
1 |
|
1 |
|
MANDATORY
1 |
|
1 |
|
SUPPORTS
1 |
|
1 |
|
NOT_SUPPORTED
1 |
|
1 |
|
NEVER
1 |
|
1 |
|
NESTED
1 |
|
1 |
|
1 |
|
NESTED & REQUIRES_NEW 区别
-
**NESTED: ** 开始一个 “嵌套的” 事务, 它是已经存在事务的一个真正的子事务。潜套事务开始执行时, 它将取得一个 savepoint. 如果这个嵌套事务失败, 我们将回滚到此 savepoint。嵌套事务是外部事务的一部分, 只有外部事务结束后它才会被提交。
-
**REQUIRES_NEW: ** 启动一个新的, 不依赖于环境的 “内部” 事务。这个事务将被完全 commited 或 rollback 而不依赖于外部事务, 它拥有自己的隔离范围, 自己的锁, 等等。当内部事务开始执行时, 外部事务将被挂起, 内务事务结束时, 外部事务将继续执行。
结论: REQUIRES_NEW 完全是一个新的事务。而 NESTED 则是外部事务的子事务, 如果外部事务 commit, 嵌套事务也会被 commit, 外部事务 rollback, 嵌套事务也会被 rollback。当嵌套事务异常回滚时不会对外部事务造成影响, 即: 外部事务可以正常提交事务.(如果外部事务 commit, 嵌套事务也会被 commit(前提: 嵌套事务没有出现异常的情况))
说明: NESTED 只是Spring针对JDBC3.0以上版本SavePoint机制的一个事务传播机制的扩展