Spring事务的传播特性决定了事务的边界如何在方法之间传递。Spring支持7种事务的传播特性:REQUIRED、SUPPORTS、MANDATORY、REQUIRES_NEW、NOT_SUPPORTED、NEVER、NESTED。这些传播行为可以控制事务的作用范围,并解决业务逻辑中方法调用时的事务问题。例如,REQUIRED表示如果当前存在事务,则加入该事务;如果没有事务,就新建一个事务。这是最常用的传播特性,因为它支持当前的业务方法在现有事务上下文中运行,确保了数据操作的一致性。
一、REQUIRED
REQUIRED 在Spring事务管理中是最常见的传播特性。如果当前存在事务,方法将会在这个事务中运行,否则,将会启动一个新的事务并在其内部执行。这种模式适用于大多数的业务场景,因为它保证了事务的连续性和数据一致性。
当一个调用的方法运行在REQUIRED传播特性中时,如果调用者方法没有事务,Spring将为被调用方法新建一个事务。如果调用者有一个事务,被调用方法则在调用者的事务环境中执行。这提供了一种顺畅的事务嵌套机制,确保了事务在业务方法间流转不会受阻。
二、SUPPORTS
SUPPORTS 特性意味着方法将在调用者事务的环境下执行。如果调用者没有开启事务,方法将在非事务的环境下执行。它适合于不需要事务控制的读操作,因为此类操作不需要修改数据,即便在没有事务的情况下也不会有一致性问题。
在使用SUPPORTS特性时,如果调用者方法有一个活动的事务,则被调用方法会在这个事务中运行。如果没有,则它在没有事务的情况下执行,不会影响性能,同时避免了不必要的事务处理。
三、MANDATORY
MANDATORY 特性要求调用方法必须运行在一个已经存在的事务中。如果调用者方法没有事务,则会抛出异常。这种传播行为是对REQUIRED特性的补充,用于那些必须在事务中执行的操作,如数据库的更新操作。
在MANDATORY环境下,如果被调用方法检测到没有活动的事务,则会抛出一个异常,这就迫使开发者确保有一个已经开启的事务。因此可以用来作为强制事务存在的一种策略。
四、REQUIRES_NEW
REQUIRES_NEW 表明被调用方法必须运行在自己的事务中,如果调用者已有事务,就将其挂起。这适用于需要完全独立于外部事务进行操作的场合,例如,在一个长事务中,一个操作不应该被外部事务的回滚影响。
REQUIRES_NEW特性能确保被调用方法总是有一个新的、独立的事务环境,这样即使发生回滚,也只会回滚该方法内的操作,而不影响外部的事务。
五、NOT_SUPPORTED
NOT_SUPPORTED 特性表明方法不应该运行在事务环境中,如果存在一个事务上下文,那么它将会被挂起。这主要用在不需要事务支持的场合,例如纯读取数据的操作。
在NOT_SUPPORTED环境中的方法被调用时,如果存在一个活动的事务,Spring会暂停当前事务,并在一个非事务性的环境中执行。这对于一些只读操作来说,可以减少不必要的开销。
六、NEVER
NEVER 特性强制规定被调用的方法不得在事务环境中执行。如果存在活动的事务,将抛出异常。基本上,它是MANDATORY的对立面,用于那些必须在没有事务的环境下执行的操作。
在NEVER传播行为下,如果方法在调用时检测到存在事务,就会直接抛出异常,从而确保其执行环境的非事务性。
七、NESTED
NESTED 特性意味着如果当前存在事务,则执行一个嵌套事务;如果当前没有事务,则其表现如REQUIRED。这是一种较为高级的事务管理方式,它允许单独回滚局部的代码段,从而不影响外层事务的完整性。
NESTED传播行为是通过SAVEPOINT来完成事务回滚的。如果一个NESTED特性的方法出现异常,只有这个子事务中的操作会被回滚,而外部事务则不受影响。这增加了对事务管理的精细控制。
相关问答FAQs:
事务的传播特性是什么?
事务的传播特性指的是在一个方法调用另一个方法时,事务如何传播到被调用的方法中。Spring框架提供了七种事务的传播特性,包括REQUIRED、SUPPORTS、MANDATORY、REQUIRES_NEW、NOT_SUPPORTED、NEVER和NESTED。
REQUIRED传播特性是如何工作的?
REQUIRED是Spring的默认事务传播特性。当一个方法被调用时,如果当前没有事务存在,则会创建一个新的事务。如果当前已经存在一个事务,则该方法将会在这个事务中运行。
REQUIRES_NEW传播特性和REQUIRED有什么不同?
REQUIRES_NEW传播特性是在每次方法调用时都创建一个新的事务。即使当前已经存在一个事务,该方法也会挂起当前事务,创建一个新的事务,并在方法结束后恢复之前的事务。
SUPPORTS传播特性是什么意思?
SUPPORTS传播特性表示如果当前存在一个事务,则方法在这个事务中运行;如果当前没有事务,则方法会以非事务的方式运行。这意味着SUPPORTS传播特性允许方法在有事务的环境和无事务的环境下运行,但不会主动创建事务。
MANDATORY传播特性有什么作用?
MANDATORY传播特性要求调用方法时必须存在一个事务。如果当前没有事务存在,则会抛出异常。这种传播特性适合于必须在已经存在事务的情况下执行的方法。
NOT_SUPPORTED传播特性的用途是什么?
NOT_SUPPORTED传播特性表示方法不应该在任何事务中运行。如果当前存在一个事务,则会将其挂起,方法在没有事务的环境下运行,执行完成后会恢复原来的事务状态。
NEVER传播特性和NOT_SUPPORTED有什么区别?
NEVER传播特性要求调用方法不能在事务中运行。如果当前存在一个事务,则会抛出异常。而NOT_SUPPORTED传播特性允许在有事务的环境下运行,但会将事务挂起。
NESTED传播特性与其他传播特性有什么不同?
NESTED传播特性表示如果当前存在一个事务,则方法在该事务的嵌套事务中运行;如果当前没有事务,则创建一个新的事务并在其中运行。与REQUIRES_NEW传播特性不同的是,NESTED传播特性可以嵌套在当前事务中,这样可以对嵌套事务进行回滚而不影响外部事务。而REQUIRES_NEW传播特性会挂起当前事务,并在方法执行完成后重新启动之前的事务。