关于 Spring 中事务方法的调用
MOKE 2020-12-06 AM loading 0条

前言
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方法 时,是无法触发代理的。

解决方案

  1. 第一种: 通过依赖注入
    创建新的 Service,将 @Transactional 标注的方法抽离出来,通过 @Autowire 注入进来进行调用。
  2. 第二种: 直接获取代理对象
    开启注解:@EnableAspectJAutoProxy(exposeProxy = true)
     Demo demoProxy = (Demo)AopContext.currentProxy();
     demoProxy.second();
  3. 第三种: 获取实例化后的 Demo
     @Autowire
     private ApplicationContext applicationContext;
     Demo demo = applicationContext.getBean(this.getClass());
     demo.second();

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

标签: Spring

非特殊说明,本博所有文章均为博主原创。

上一篇 Oracle与MySQL的学习
下一篇 没有了

评论啦~


召唤看板娘