Spring事务调用类自己方法失效解决办法和原因

时间:2023-03-09 17:37:34
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());