Spring事务调用类自己方法失效解决办法和原因
问题
正常情况下,我们都是在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());