Spring 中事务方法的调用

前言
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 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();

其实,这几种方法都是为了调用到代理子类中的方法

你可能感兴趣的:(java基础,Spring,java,jar,apache)