时刻感觉自己要写出bug的节奏,瑟瑟发抖。
最近开发中发现一个问题,在使用@Transactional注解时,发现注解未生效,注解的方法在出现异常时,并没有回滚。因此搜了一些该注解失效的原因。
@Transactional注解无效的原因
1. 被注解的方法不是public的
2. 异常类型不是unchecked类型。如果checked异常也想被回滚,注解上写明异常类型。
@Transactional(rollbackFor=Exception.class)
3. 数据库引擎需要支持事务。
4. 是否开启了对注解的解析。
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>
5. spring是否扫描到这个包。
6. 检查是不是在同一个类的方法中调用。
自己代码中失效的原因
被注解的方法是public的,注解上也写明了异常类型,数据库引擎是innodb。第四点暂时没有找到在哪里配置。该方法所在的包也是被扫描到的。异常也没有catch。
最终在我这边,该注解失效的原因是在同一个类中的方法调用。
失效原理分析
在同一个类中调用方法失效的原因是,事务管理是基于动态代理对象的代理逻辑实现的,那么如果在类内部调用类内部的事务方法,这个调用事务方法的过程并不是通过代理对象来调用的,而是直接通过this对象来调用方法,绕过的代理对象,肯定就是没有代理逻辑了。
参考: