Spring中事务的传播行为有哪些?
在 Spring 框架里,事务传播行为用于定义在嵌套事务场景下,新事务与现有事务之间的交互方式。Spring 定义了 7 种事务传播行为,这些行为由org.springframework.transaction.annotation.Propagation
枚举类提供。以下是对这些传播行为的详细介绍:
1. Propagation.REQUIRED
(默认值)
- 基本含义:这是最常用的事务传播行为。当一个带有
REQUIRED
传播行为的方法被调用时,如果当前已经存在一个事务,那么该方法会加入到这个现有的事务中执行;如果当前没有事务,那么它会创建一个新的事务来执行自身的操作。 - 实际场景示例:假设你有一个电商系统,有一个
OrderService
类的createOrder
方法和一个InventoryService
类的decreaseInventory
方法。createOrder
方法用于创建订单,decreaseInventory
方法用于减少商品库存。createOrder
方法开启了一个事务,在其内部调用了decreaseInventory
方法,由于decreaseInventory
方法默认传播行为是REQUIRED
,它会加入到createOrder
方法的事务中。这样,如果在整个过程中出现异常,整个订单创建和库存减少的操作都会回滚,保证了数据的一致性。
2. Propagation.SUPPORTS
- 基本含义:当使用
SUPPORTS
传播行为时,如果当前存在事务,该方法会加入到这个事务中执行;如果当前没有事务,那么它会以非事务的方式执行。也就是说,这个传播行为对事务是 “支持” 的,但不是强制要求必须在事务中执行。 - 实际场景示例:有一个
UserService
类的getUserInfo
方法,这个方法只是用于查询用户信息,通常情况下不需要事务。但如果在一个已经存在事务的方法中调用了getUserInfo
方法,它可以加入到这个事务中。比如在一个处理用户订单的事务中,需要查询用户的基本信息来完善订单,此时getUserInfo
方法就可以加入到这个订单处理的事务中。
3. Propagation.MANDATORY
- 基本含义:
MANDATORY
传播行为要求当前必须存在一个事务,该方法才能执行。如果当前没有事务,会抛出异常。它强调了事务的强制性。 - 实际场景示例:在一些涉及到重要数据更新且必须保证原子性的操作中可以使用。例如,一个金融系统中的
TransferService
类的transferMoney
方法用于进行资金转账操作,这个操作必须在事务中进行,因为涉及到资金的增减。如果在调用这个方法时没有事务,就会抛出异常,防止出现数据不一致的情况。
4. Propagation.REQUIRES_NEW
- 基本含义:无论当前是否存在事务,
REQUIRES_NEW
传播行为都会创建一个新的事务来执行该方法。如果当前已经存在一个事务,那么会将当前事务挂起,等新事务执行完毕后,再恢复原来的事务继续执行。 - 实际场景示例:假设你有一个日志记录的方法
LogService
类的recordLog
方法,即使主业务方法出现异常回滚,日志记录也应该是独立的,不应该受到主业务事务的影响。此时可以将recordLog
方法的传播行为设置为REQUIRES_NEW
,这样它会创建一个新的事务来记录日志,与主业务事务相互独立。
5. Propagation.NOT_SUPPORTED
- 基本含义:使用
NOT_SUPPORTED
传播行为的方法会以非事务的方式执行。如果当前存在事务,会将当前事务挂起,等该方法执行完毕后,再恢复原来的事务继续执行。 - 实际场景示例:一些对性能要求较高且不需要事务保证的操作可以使用这种传播行为。比如一个
CacheService
类的updateCache
方法,用于更新缓存数据,这个操作本身不需要事务,而且如果在一个事务中进行可能会影响性能。当在一个事务方法中调用updateCache
方法时,会将事务挂起,以非事务方式更新缓存。
6. Propagation.NEVER
- 基本含义:
NEVER
传播行为要求该方法必须以非事务的方式执行。如果当前存在事务,会抛出异常。它与MANDATORY
相反,是坚决不允许在事务中执行。 - 实际场景示例:对于一些简单的查询操作,并且明确不希望在事务中执行的情况可以使用。比如一个
StatisticsService
类的getStatisticsData
方法,只是从数据库中查询统计数据,不需要事务保证,使用NEVER
传播行为可以确保它不会在事务中执行。
7. Propagation.NESTED
- 基本含义:如果当前存在事务,
NESTED
传播行为会在当前事务中创建一个嵌套事务。嵌套事务是当前事务的一个子事务,有自己独立的保存点。如果嵌套事务执行失败,可以只回滚嵌套事务,而不影响外部事务;如果外部事务回滚,嵌套事务也会回滚。如果当前没有事务,它会创建一个新的事务,和REQUIRED
类似。 - 实际场景示例:在一个复杂的业务流程中,有一个主业务方法
MainService
类的mainBusiness
方法,其中包含了一些子操作,这些子操作可以使用NESTED
传播行为。比如在主业务中处理订单时,可能会有一个子操作是发送通知邮件,这个发送邮件的操作可以作为一个嵌套事务。如果发送邮件失败,只回滚这个子事务,不影响整个订单处理的主事务。