springboot事务不生效的几种解决方案
最近在业务中遇到了很奇怪的场景,在加上 @Transactional注解后,插入2条数据,如果第二条出异常了,第一条不会回滚,排查了很久,上网也找资料看了很久,问题得以解决,总结了事务不生效的几点可能的原因:
1.mysql的MyISAM引擎不支持回滚,如果需要自动回滚事务,需要将mysql的引擎设置成InnoDB;
2.在业务中抛出异常时,本应该被事务管理器捕获的异常,
被手动catch处理了,或者事务结果未满足具体业务需求的,如果需要手动catch异常做业务处理,需要在catch里手动回滚事务TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(),
或者在catch中主动抛出异常throw new RuntimeException();.
3.默认的spring事务只会捕获RuntimeException,如果是非运行时异常也需要进行事务回滚的话,可以在@Transactional注解中加上rollbackfor = Exception.class属性;
项目中没有配置事务管理器(大坑!我遇到的就是这个),需要在配置类或者配置文件中配置,以本人的配置类为例,因为项目是多数据源的,所以要区别配置不同数据源的事务管理器.
数据源一:
public DataSource getDataSource() {
return createDataSource();
}
@Bean(name = "detTransactionManager")
public PlatformTransactionManager txManager(@Qualifier("detDataSource") DataSource dataSource) {
return new DataSourceTransactionManager(dataSource);
}
数据源二:
@Bean(name = "shardDataSource")
public DataSource getDataSource() {
return buildDataSource();
}
@Bean(name = "shardTransactionManager")
public PlatformTransactionManager txManager(@Qualifier("shardDataSource") DataSource dataSource) {
return new DataSourceTransactionManager(dataSource);
}
可以看到,两个事务管理器配置了不同的beanName,接下来只需要 在需要事务控制的位置加上该事务管理器的name就可以完美解决啦!
@Override
@Transactional(value = "detTransactionManager",rollbackFor = Exception.class)
public int updateOrInsert(BaseRequest param) {