问题

正常情况下,我们都是在controller里调用service里的方法,这个方法如果需要加事务,就在方法上加上@Transactional,这样是没问题的,事务会生效。

可是如果像下面这样,绕以下,service里方法调用了自己类的方法,这个时候即使加了@Transactional,事务也不会生效。

@Controller
public class TestController {
    
    @Autowired
    private TestService testService;
    
    @GetMapping("/test")
    public void test(){
        testService.methodB();
    }
    
}
@Service
public class TestService {

    @Transactional
    public void methodA(){

    }

    /**
     * 这里调用methodA() 的事务将会失效
     */
    public void methodB(){
        this.methodA();
    }

}

原因

因为,spring的事务实现是使用了代理类来实现,而这里的this.methodA(),并没有走TestService的代理类,所以事务会失效。

解决

方法1:把methodA()和methodB()分别放到不同的类里。

方法2:自己注入自己,用注入的实例调用

@Service
public class TestService {

    @Autowired
    private TestService testService;
    
    @Transactional
    public void methodA(){

    }

    /**
     * 这里调用methodA() 的事务将会生效
     */
    public void methodB(){
        testService.methodA();
    }

}

方法3:获取代理类,利用代理类调用自己类的方法

@Service
public class TestService {

    @Transactional
    public void methodA(){

    }

    /**
     * 这里调用methodA() 的事务将会生效
     */
    public void methodB(){
        ((TestService)AopContext.currentProxy()).methodA();
    }

}

最后

spring的aop代理有jdk代理和cglib代理实现,具体参阅java中代理,静态代理,动态代理以及spring aop代理方式,实现原理统一汇总

        //是否代理对象
        AopUtils.isAopProxy(AopContext.currentProxy());

        //是否cglib 代理对象
        AopUtils.isCglibProxy(AopContext.currentProxy());

        //是否jdk动态代理
        AopUtils.isJdkDynamicProxy(AopContext.currentProxy());