手动事务提交

我们现在进行事务控制一般都是使用注解型事务,然而有些时候某些项目注解型事务却失效了。比如下方的这种情况:

@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中这样问题,应该还是整合的问题,这里就不详细探讨了。

你可能感兴趣的:(手动回滚事务)