在使用Spring管理事务时会遇到一些情况使事务失效,下面列举一些常见的情况:
1. 数据库存储引擎不支持
*本人因为这个问题整整白活1小时*
在MySQL中,在MySQL 5.5.5版本之前默认的存储引擎为MyISAM,但是MyISAM引擎是不支持事务的,InnoDB才支持事务。可能有一些特殊情况导致使用MySQL 5.5.5之后版本的默认的存储引擎也为MyISAM,如果是这样,那么Spring事务便不会生效。可以在MySQL查询控制台中使用show engines;
命令查看默认的存储引擎,比如:
我这里现在默认的存储引擎为MyISAM,可以在MySQL的配置文件中修改新建表的默认引擎,当然,也可以修改表的引擎为InnoDB,如:
或使用命令:
代码语言:javascript复制alter table tb_account
engine =InnoD;
2. @Transactional
注解所在的类不是 Spring
容器的 bean
代码语言:javascript复制@Service
public class LogServiceImpl implements LogService {
@Transactional
public void log(String out, String in, Double money) {
...
};
};
此时,如果把 @Service
注解删除,那么这个类就不会被加载成一个 bean
,那么这个类不会被 Spring
管理了,事务失效
3. 方法修饰符非 public
@Transactional
注解只能用于修饰符为 public
的方法上,否则事务会失效;即使方法的修饰符为 public
,但是如果被修饰符为 private
的方法调用,事务同样会失效
4. 数据源配置事务管理器有误
代码语言:javascript复制@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {
DataSourceTransactionManager transactionManager = new DataSourceTransactionManager();
transactionManager.setDataSource(dataSource);
return transactionManager;
};
假如项目使用MyBatis管理,事务管理器的数据源DataSource
必须和MyBatis中的DataSource
一致,并且事务管理器要被Spring管理。如果没有配置事务管理器或者事务管理器配置有误,事务失效
5. catch
语句未抛出异常
程序异常被忽略,并且不抛出异常,事务会失效
6. 抛出的异常类型错误
@Transactional
默认回滚的异常是 RuntimeException
和 Error
,假如不进行设置,如果遇到Exception
异常,事务会失效