SpringBoot @Transactional事务不生效排查方式
作者:magic_kid_2010
问题描述
在开发Spring Boot项目时,方法使用了@Transactional注解,但是当方法抛出异常时,数据没有回滚。
解决方法
经排查,发现是数据库引擎搞的鬼。
之前在建mysql数据库表的时候,忘记修改表使用的引擎,引擎使用的是MyISAM,而MyISAM是不支持事务的,只需将引擎改为InnoDB即可。
排查步骤
当@Transactional事务不生效的时候,可以遵循以下排查步骤:
1、查看数据库或表,设置的引擎。MyISAM是不支持事务的,必须改为InnoDb。
2、@Transactional注解的方法是否为public。如果应用在protected、private或者 package的方法上,不会报错,但是事务设置不会起作用。
3、@Transactional所注解的方法所在的类,是否已经使用了注解@Service或@Component等。
4、需要调用该方法,且需要支持事务特性的调用方,是在 @Transactional所在的类的外面。
注意:类内部的其他方法调用这个注解了@Transactional的方法,事务是不会起作用的。
5、注解为事务范围的方法中,事务的回滚仅仅对于unchecked的异常有效。对于checked异常无效。也就是说事务回滚仅仅发生在出现RuntimeException或Error的时候。
那么什么是checked异常呢?
java里面将派生于Error或者RuntimeException(比如空指针,1/0)的异常称为unchecked异常,其他继承自java.lang.Exception得异常统称为Checked Exception,如IOException、TimeoutException等
通俗一点就是:
- 代码中出现的空指针等异常,会被回滚。
- 而文件读写、网络超时问题等,spring就没法回滚了。
unchecked异常,回滚解决办法
方法一
手动回滚。
给注解加上参数如:
@Transactional(rollbackFor=Exception.class)
方法二
如上述分析。
对uncheck异常进行catch,然后抛出RuntimeException异常。
方法三
在service层方法的catch语句中增加:
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
语句,手动回滚,这样上层就无需去处理异常。
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。