一 问题背景
前段时间在项目中使用到spring事务注解功能,在自测代码时发现异常下未能产生回滚操作,借此空闲将之前问题定位思路整理如下,希望对大家定位该类问题提供解决思路。
二 分析与解决
本次测试工程采用springBoot+mybits+mysql进行验证,在controller中注入service调用service的实现类进行事务操作,项目结构如下图所示
1. 内部调用导致事务失效
导致该问题原因为spring执行方法update的时候会生成一个动态代理类去执行代理类的该方法,代理类在执行updateUser时,执行的是当前实例的updateUser方法不会通过动态代理去进行事务增强处理所以会导致事务消除。
解决此问题需要在当前类中注入当前类的实例bean,通过当前类的实例去调用事务增强的updateUser方法,或者直接将事务添加至update方法,如下图所示:
将事务添加至直接调用方法
注入当前类实例
2. 异常捕获导致事务失效
(1)自动回滚
需要在异常处理中抛出事务所对应的异常,事务处理的异常默认为RunTimeException
(2)手动回滚
需要在异常处理中手动进行异常回滚操作,如下图所示
(3)回滚部分异常
3. 方法权限限制引起失效
spring的事务注解@Transactional只能放在public修饰的方法上才起作用,如果放在其他非public(private,protected)方法上,虽然不报错,但是事务不起作用
三 参考资料
SpringBoot2异常处理之用try/catch错误信息并回滚事务(自动回滚/手动回滚/部分回滚)