我们现在进行事务控制一般都是使用注解型事务,然而有些时候某些项目注解型事务却失效了。比如下方的这种情况:
@Transactional(rollbackFor = Exception.class)
@Override
public void batchInsertUserInfo(List<User> users){
userMapper.batchInsert(users);
}
一般来说,在批量插入出错的时候会执行回滚操作,然而事实上,事务失效了,在出错的时候没有执行回滚操作。
打印的日志中有这样的信息:
2020-04-22 09:43:46,467 DEBUG http-apr-8081-exec-9 org.mybatis.spring.SqlSessionUtils 97-> Creating a new SqlSession
2020-04-22 09:43:46,468 DEBUG http-apr-8081-exec-9 org.mybatis.spring.SqlSessionUtils 148-> SqlSession [org.apache.ibatis.session.defaults.DefaultSqlSession@3cd47914] was not registered for synchronization because synchronization is not active
2020-04-22 09:43:46,470 DEBUG http-apr-8081-exec-9 org.mybatis.spring.transaction.SpringManagedTransaction 87-> JDBC Connection [oracle.jdbc.driver.T4CConnection@4afa9054] will not be managed by Spring
JDBC Connection ** will not be managed by Spring ,事务也没有起作用。
寻思注解不行,那就用手动提交事务的方式呗。
public void batchInsertUserInfo(List<User> users){
//获取事物管理器
DataSourceTransactionManager transactionManager=SpringContextUtil.getBean(DataSourceTransactionManager.class);
DefaultTransactionDefinition def = new DefaultTransactionDefinition();
def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRES_NEW); // 事物隔离级别,开启新事务.
TransactionStatus status = transactionManager.getTransaction(def); // 获得事务状态
try{
userMapper.batchInsert(users);
transactionManager.commit(status);
}catch(Exception e){
log.error("batchInsert user fail",e)
transactionManager.rollback(status);
}
}
尴尬的事情发生了,事务竟然还是不起作用,依然打印上边的日志。
那没办法了,只能找找原因了,看看配置文件是否有问题,这个是ssm框架的web项目,发现配置文件中事务的配置没有问题。
接着排查,原来刚才检查的配置文件并没有引入到总配置文件中。
又不对了,配置文件没有生效,数据库等的配置又是怎么生效的呢。
思考是不是通过@Configuration配置类配置的,排查了下,果然发现是事务管理器是通过配置类配置的。
@Configuration
public class DataSourceConfig {
@Bean(name = "dataSource")
@Primary
public DataSource dataSource() {
DruidDataSource dataSource=new DruidDataSource();
//getDataSourceInfo()读取dataSource配置,自定义的,不需要关注
DataSourceInfo application=getDataSourceInfo();
BeanUtils.copyProperties(application,dataSource);
return dataSource;
}
@Bean(name = "transactionManager")
@Primary
public DataSourceTransactionManager memberDb1TransactionManager(
@Qualifier("dataSource") DataSource dataSource) {
return new DataSourceTransactionManager(dataSource);
}
}
按照配置类的方法,思考可以采用注入的方式来引入事务管理器transactionManager,之前的代码略作修改。
//注入DataSourceConfig下的事务管理器bean
@Resource
DataSourceTransactionManager transactionManager;
public void batchInsertUserInfo(List<User> users){
DefaultTransactionDefinition def = new DefaultTransactionDefinition();
def.setName("authOrgForCssTrans");
def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
TransactionStatus status = transactionManager.getTransaction(def);
try{
userMapper.batchInsert(users);
transactionManager.commit(status);
}catch(Exception e){
log.error("batchInsert user fail",e)
transactionManager.rollback(status);
}
}
这样一来使用手动事务控制,事务不能回滚的问题就解决了。
奇怪的是同样的配置方式,在springboot项目中并发现没有web中这样问题,应该还是整合的问题,这里就不详细探讨了。