@Transactional事物失效解决

这么一个类

public class Foo {
    @Transactional
    public void bar() { /* … */ }
 
    public void baz() {
        this.bar();
    }
}

可能会有不少人会跟我一样,觉得上面这种方式调用 baz()方法时,bar()上的@Transactional注解还是会起作用的,即bar()在被调用时,将会开启事务。 但是,当实际操作之后,你会发现,这样并不会开启新的事务?
为什么呢?
我们知道,Spring之所以可以对开启@Transactional的方法进行事务管理,是因为Spring为当前类生成了一个代理类,然后在执行相关方法时,会判断这个方法有没有@Transactional注解,如果有的话,则会开启一个事务。
但是,上面这种调用方式时,在调用baz()时,使用的并不是代理对象,从而导致this.bar()时也不是代码对象,从而导致@Transactional失败。

解决方式

1、非springboot项目 在spring的xml中加上如下配置

springboot项目 在SpringBoot 入口类中添加 @EnableAspectJAutoProxy(exposeProxy = true) 注解

@SpringBootApplication
@EnableAspectJAutoProxy(exposeProxy = true)
public class App {
 
    public static void main(String[] args) {
        SpringApplication.run(App.class);
    }
 
}

2、添加依赖

 <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-aspects</artifactId>
      <version>5.1.2.RELEASE</version>
 </dependency>

3、调用方式调整为((FOO) AopContext.currentProxy()).bar();

public class Foo {
   @Transactional
   public void bar() { /* … */ }
 
    public void baz() {
        ((Foo) AopContext.currentProxy()).bar();
    }
}

还可以将本类作为一个bean注入,然后再调用也可以开启事物,

你可能感兴趣的:(@Transactional事物失效解决)