前言
Spring 提供了事务的管理机制,我们只需要在方法或者类上加上 @Transactional 注解进行事务管理。而非事务方法与事务方法之间相互调用,有时会使事务失效,本文是对该情形下的事务总结。
什么是事务传播机制
事务在多个方法的调用中是如何传递的,是重新创建事务还是使用父方法的事务?父方法的回滚对子方法的事务是否有影响?这些都是可以通过事务传播机制来决定的。
Spring 中定义了7中事务传播机制:
- PROPAGATION_REQUIRED
支持当前事务,如果当前没有事务,就新建一个事务。 - PROPAGATION_SUPPORTS
支持当前事务,如果当前没有事务,就以非事务方式执行。 - PROPAGATION_MANDATORY
支持当前事务,如果当前没有事务,就抛出异常。 - PROPAGATION_REQUIRES_NEW
新建事务,如果当前存在事务,把当前事务挂起。 - PROPAGATION_NOT_SUPPORTED
以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。 - PROPAGATION_NEVER
以非事务方式执行,如果当前存在事务,则抛出异常。 - PROPAGATION_NESTED
如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作
Spring 默认是 PROPAGATION_REQUIRED 机制。 下面来举一个例子:
public class Demo
{
@Transactional
public void fisrt(){
second();
}
public void second(){}
}
这种情况,first 方法标注了 @Transactional 注解,而 second 方法为普通方法。由于 REQUIRED 机制,执行的时候,事务会传播给 second 方法,因此执行到 second 方法时,事务依然是生效的。接下来,反过来,就是本次遇到的问题:
public class Demo
{
public void fisrt(){
second();
}
@Transactional
public void second(){}
}
第二种情况,即使 second方法标注了 @Transactional 注解,事务也不会生效。
原因: 这是因为 Spring 采用动态代理机制来实现事务控制。在扫描 Bean 的时候,会给有 @Transactional 的类,生成一个代理子类。只有调用代理子类对象的方法,才会新建一个事务并调用原始对象的方法。而第二种情况,直接在原始对象 Demo 中调用标注了 @Transactional 注解的 second方法 时,是无法触发代理的。
解决方案
- 第一种: 通过依赖注入
创建新的 Service,将 @Transactional 标注的方法抽离出来,通过 @Autowire 注入进来进行调用。 - 第二种: 直接获取代理对象
开启注解:@EnableAspectJAutoProxy(exposeProxy = true)Demo demoProxy = (Demo)AopContext.currentProxy(); demoProxy.second();
- 第三种: 获取实例化后的 Demo
@Autowire private ApplicationContext applicationContext; Demo demo = applicationContext.getBean(this.getClass()); demo.second();
其实,这几种方法都是为了调用到代理子类中的方法。