Spring @Transactional,具有跨多个数据源的事务

时间:2018-02-23 19:15:32

标签: spring hibernate transactions spring-data-jpa transactionmanager

我必须将两个数据源更新为一个事务的一部分。那是 -

  1. 我在DB1中进行了更新。
  2. 然后,我在DB2中再做一次更新。
  3. 如果DB2中的更新失败,我想回滚DB1和DB2以进行回滚。这可以使用@Transactional完成吗?

    以下是示例代码 -

    @Transactional(value="db01TransactionManager")
    public void updateDb01() {
        Entity01 entity01 = repository01.findOne(1234);
        entity01.setName("Name");
        repository01.save(entity01);
    
        //Calling method to update DB02
        updateDb02();
    }
    
    @Transactional(value="db02TransactionManager")
    public void updateDb02() {
        Entity02 entity02 = repository02.findOne(1234);
        entity02.setName("Name");
        repository02.save(entity02);
    
        //Added this to force a roll back for testing
        TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
    }
    

    我的问题是,updateDb02中的setRollbackOnly()仅回滚Db01事务。

3 个答案:

答案 0 :(得分:6)

我使用ChainedTransactionManager - http://docs.spring.io/spring-data/commons/docs/1.6.2.RELEASE/api/org/springframework/data/transaction/ChainedTransactionManager.html

解决了这个问题

Spring Boot配置:

    @Bean(name = "chainedTransactionManager")
    public ChainedTransactionManager transactionManager(@Qualifier("primaryDs") PlatformTransactionManager ds1,
                                                    @Qualifier("secondaryDs") PlatformTransactionManager ds2) {
         return new ChainedTransactionManager(ds1, ds2);
    }

然后您可以按如下方式使用它:

@Transactional(value="chainedTransactionManager")
public void updateDb01() {
    Entity01 entity01 = repository01.findOne(1234);
    entity01.setName("Name");
    repository01.save(entity01);

    //Calling method to update DB02
    updateDb02();
}

public void updateDb02() {
    Entity02 entity02 = repository02.findOne(1234);
    entity02.setName("Name");
    repository02.save(entity02);

    //Added this to force a roll back for testing
    TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}

答案 1 :(得分:1)

最好的方法是创建第三个方法,该方法将注释为@Transactional

@Transactional(readOnly = false)
public void updateCommon(){
  upbateDb01();
  upbateDb02();
}

根据spring文档,当第一个注释出现时,事务控制开始,因此在这种情况下,将在调用updateCommon时启动单个事务。 的更新 但是如果您使用CrudRepository或类似的东西,这将有效。

如果有多个数据源,您可以尝试使用全局事务管理概念。以下是spring文档中的示例:

@Inject private PlatformTransactionManager txManager; 

TransactionTemplate template  = new TransactionTemplate(this.txManager); 
template.execute( new TransactionCallback<Object>(){ 
  public void doInTransaction(TransactionStatus status){ 
   // work done here will be wrapped by a transaction and committed. 
   // the transaction will be rolled back if 
   // status.setRollbackOnly(true) is called or an exception is thrown 
  } 
});

这是一个链接:http://spring.io/blog/2011/08/15/configuring-spring-and-jta-without-full-java-ee/ 我从来没有把它用于我自己,所以我没有深入探讨这个话题。 希望它会有所帮助

答案 2 :(得分:0)

我相信你已经定义了如下所示的txns。

@Bean(name="db01TransactionManager") 
@Autowired
DataSourceTransactionManager tm1(@Qualifier ("datasource1") DataSource datasource) {
    DataSourceTransactionManager txm  = new DataSourceTransactionManager(datasource);
    return txm;
}

@Bean(name="db02TransactionManager") 
@Autowired
DataSourceTransactionManager tm2(@Qualifier ("datasource2") DataSource datasource) {
    DataSourceTransactionManager txm  = new DataSourceTransactionManager(datasource);
    return txm;
}

现在最简单的方法是尝试,捕获和回滚两个事务。但如果您仍想委托,则可以选择以下选项。

创建自己的并覆盖回滚方法并使用它。

@Bean(name=“allTransactionManager") 
@Autowired
DataSourceTransactionManager tm2(@Qualifier ("datasource1”) DataSource datasource1, @Qualifier ("datasource2") DataSource datasource2) {

    DataSourceTransactionManager txm  = new MyDataSourceTransactionManager(datasource1,datasouce2);
        return txm;
}

并将您自己的交易管理员定义为。

MyDataSourceTransactionManager extends DataSourceTransactionManager{
DataSourceTransactionManager tm1; 
DataSourceTransactionManager tm2; 

MyDataSourceTransactionManager(Datasource ds1,Datasource d2){
  tm1 = new DataSourceTransactionManager(DataSource);
  tm2 =new DataSourceTransactionManager(DataSource);
}
// override and for roll back, rollback for both of tm1 and tm2. Thus all actions are delegated in this class

}

然后在你希望同步工作的情况下将它用于dao图层。

 @Transactional("allTransactionManager")

所以现在我们有了自己的事务管理器,它能够为两种类型的事务一起回滚或提交。