事务在大多数面试中都会被问到。假如问到Spring,也一般会问到Spring的隔离机制和传播行为。隔离机制在第一篇的博客中已经介绍过了,并且Spring的隔离机制依旧采用数据库的四种。下面简单的介绍一下Spring的七种传播行为。
Spring的传播行为定义在TransactionDefinition这个接口中。
1、PROPAGATION_REQUIRED
这是默认的属性。如果存在一个事务,则支持当前事务。如果没有事务,就开启一个新的事务。
2、PROPAGATION_MANDATORY 支持当前事务,如果没有事务,就抛出异常。
3、PROPAGATION_NERER 以非事务的方式执行,如果当前存在事务,就抛出异常。
4、PROPAGATION_NOT_SUPPORTED 以非事务的方式执行,如果当前存在事务,就把当前事务挂起。
5、PROPAGATION_REQUIRES_NEW 新建事务,如果当前存在事务,就把当前事务挂起。
6、PROPAGATION_SUPPORTS 支持当前事务,如果当前没有事务,就以非事务的方式运行。
7、PROPAGATION_NESTED 支持当前事务,新增Savepointed点,与当前事务同步提交或执行。
注意:嵌套事务一个很重要的概念就是内层事务依赖外层事务。
PROPAGATION_REQUIRES_NEW 这个不是嵌套事务,是两个相互独立的事务。一旦内层事务提交后,外层事务不能对其进行回滚。两个事务互不影响,同时需要JTA事务管理器的影响。
PROPAGATION_NESTED
这是是真正的嵌套事务。也就是说,外层事务失败时,会回滚内层事务所做的操作。内层事务失败时,并不会回滚外层事务所做的操作。
举例:
void methodA(){} //Propagation_Required
void method_Required(){}
void method_Mandatory(){}
void
method_Requires_New(){}
void method_Supports(){}
void method_Nested(){}
PROPAGATION_REQUIRED:调用methodA()时,环境中没有事务,所以开启一个事务。当在methodA()中调用method_Required()时,环境中已经有事务了。所以,method_Required()就加入当前事务。
PROPAGATION_MANDATORY:单独调用method_Mandatory()方法时,因为当前没有一个活动的事务,则会抛出IllegalTransactionStateException异常,当调用methodA()时,method_Mandatory()则被加入到了methodA()的事务中。
PROPAGATION_REQUIRES_NEW:如果methodA()方法在调用method_Requires_New()方法失败了,如果method_Requires_New()执行成功,则method_Requires_New()方法所有的结果依然会被会被提交。而除了method_Requires_New()之外的其他代码都会被回滚。在method_Requires_New()执行时,methodA()会被挂起,当method_Requires_New()执行完成之后,methodA()将继续执行。
PROPAGATION_SUPPORTS:单纯调用method_Mandatory()时,该方法是以非事务执行的。当调用methodA()时,method_Mandatory()就加入了methodA()的事务中。
PROPAGATION_NESTED:当method_Nested()方法被调用之前,会保存当前状态到java.sql.Savepoint中。如果method_Nested()方法调用失败,则恢复到之前保存的状态。这是的事务并没有提交,如果后续的代码调用失败,则回滚包括method_Nested()方法的所有操作。外层事务失败,会回滚内层事务的所有操作。而内层事务操作失败并不会引起完成事务的回滚。这点就是与Propagation_Requires_New最主要的区别。