问题
正常情况下,我们都是在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());
|