前言 Spring 提供了事务的管理机制,我们只需要在方法或者类上加上 @Transactional 注解进行事务管理。而非事务方法与事务方法之间相互调用,有时会使事务失效,本文是对该情形下的事务总结 什么是事务传播机制 事务在多个方法的调用中是如何传递的,是重新创建事务还是使用父方法的事务?父方法的回滚对子方法的事务是否有影响?这些都是可以通过事务传播机制来决定的。
Spring 默认是 PROPAGATION_REQUIRED 机制。 下面来举一个例子:
public class Demo
{
@Transactional
public void fisrt(){
second();
}
public void second(){}
}
这种情况,first 方法标注了 @Transactional 注解,而 second 方法为普通方法。由于 REQUIRED 机制,执行的时候,事务会传播给 second 方法,因此执行到 second 方法时,事务依然是生效的。接下来,反过来,就是本次遇到的问题:
public class TestService
{
public void fisrt(){
second();
}
@Transactional
public void second(){}
}
第二种情况,即使 second方法标注了 @Transactional 注解,事务也不会生效。 原因: 这是因为 Spring 采用动态代理机制来实现事务控制。在扫描 Bean 的时候,会给有 @Transactional 的类,生成一个代理子类。只有调用代理子类对象的方法,才会新建一个事务并调用原始对象的方法。而第二种情况,直接在原始对象 TestService 中调用标注了 @Transactional 注解的 second方法 时,是无法触发代理的。spring的事务实现是使用了代理类来实现,而这里的second(),并没有走TestService的代理类,所以事务会失效。
@Service
public class OrderServiceImpl implements OrderService {
@Transactional
public void update(Order order) {
updateOrder(order);
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void updateOrder(Order order) {
}
}
这次在 update 方法上加了 @Transactional,updateOrder 加了 REQUIRES_NEW 新开启一个事务,那么新开的事务管用么? 这两个例子的答案是:不管用! 因为它们发生了自身调用,就调该类自己的方法,而没有经过 Spring 的代理类,默认只有在外部调用事务才会生效,这也是老生常谈的经典问题了。
解决方案: 第一种: 通过依赖注入 创建新的 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();
其实,这几种方法都是为了调用到代理子类中的方法。
|