1

我正在尝试将使用 JDBC 模板的 Spring Boot 项目版本 2.3.0.M3 迁移到 R2DBC。该项目还使用 Liquibase,因此我无法完全摆脱 JDBC。我在项目中同时拥有 spring-boot-starter-data-r2dbc 和 spring-boot-starter-jdbc 依赖项,在尝试运行我的一项测试时出现以下异常:

org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'org.springframework.transaction.TransactionManager' available: expected single matching bean but found 2: transactionManager,connectionFactoryTransactionManager

    at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveNamedBean(DefaultListableBeanFactory.java:1180)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveBean(DefaultListableBeanFactory.java:416)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:349)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:342)
    at org.springframework.transaction.interceptor.TransactionAspectSupport.determineTransactionManager(TransactionAspectSupport.java:480)
    at org.springframework.transaction.interceptor.TransactionAspectSupport.invokeWithinTransaction(TransactionAspectSupport.java:335)
    at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:99)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:186)
    at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.proceed(CglibAopProxy.java:747)
    at org.springframework.aop.interceptor.ExposeInvocationInterceptor.invoke(ExposeInvocationInterceptor.java:95)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:186)
    at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.proceed(CglibAopProxy.java:747)
    at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:689)
...

bean connectionFactoryTransaction 管理器在 Spring 类 R2dbcTransactionManagerAutoConfiguration 中定义如下:

    @Bean
    @ConditionalOnMissingBean(ReactiveTransactionManager.class)
    public R2dbcTransactionManager connectionFactoryTransactionManager(ConnectionFactory connectionFactory) {
        return new R2dbcTransactionManager(connectionFactory);
    }

bean transactionManager 在 Spring 类 DataSourceTransactionManagerAutoConfiguration 中是这样定义的:

   @Bean
   @ConditionalOnMissingBean(PlatformTransactionManager.class)
   DataSourceTransactionManager transactionManager(DataSource dataSource,
           ObjectProvider<TransactionManagerCustomizers> transactionManagerCustomizers) {
       DataSourceTransactionManager transactionManager = new DataSourceTransactionManager(dataSource);
       transactionManagerCustomizers.ifAvailable((customizers) -> customizers.customize(transactionManager));
       return transactionManager;
   }

可以看出,@ConditionalOnMissingBean 注释包含不同的类型,这将导致创建两个 bean 的实例。但是,在 Spring 类 TransactionAspectSupport 中,determineTransactionManager 方法中有这行代码:

defaultTransactionManager = this.beanFactory.getBean(TransactionManager.class);

由于 DataSourceTransactionManager 和 R2dbcTransactionManager 这两种事务管理器类型都实现了 TransactionManager 接口,因此上述两个事务管理器 bean 都会匹配并发生错误。

我现在正在联系是否有人设法解决或解决此问题?
提前致谢!

4

2 回答 2

1

受 M. Deinums 回答(谢谢!)的启发,我将以下步骤应用于我的项目,之前失败的测试现在成功运行:

  • 删除 spring-boot-starter-jdbc 依赖项。
  • 将依赖项添加到 spring-jdbc。
  • 将依赖项添加到 HikariCP (com.zaxxer)。
  • 添加 spring.liquibase 用户和密码属性(我已经有了 url 和 change-log 属性)。
  • 删除所有 spring.datasource 属性(我有 url 和 drive-class-name)。

我定义了不需要更改的 spring.r2dbc 属性用户名、密码和 url。

更新:
另外,我在测试中使用了 Testcontainers 并且无法分配静态端口。为了能够在 Liquibase 上配置数据库端口,我覆盖了 SpringLiquibase 类型的 bean 名称 liquibase,并在 liquibase bean 创建方法中创建了一个 DataSource(未公开为 bean)并将其设置在 liquibase bean 上。

于 2020-03-25T13:37:01.957 回答
1

有可能spring-boot-starter-jdbc并存spring-boot-starter-data-r2dbc。有一个类org.springframework.transaction.annotation.TransactionManagementConfigurer可以用来解决冲突。当 r2dbc 存在时,Spring Boot 2.3.0 似乎禁用了自动数据源配置。可以手动导入org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration类以使两者共存。

@Bean
TransactionManagementConfigurer transactionManagementConfigurer(ReactiveTransactionManager reactiveTransactionManager) {
    return new TransactionManagementConfigurer() {
        @Override
        public TransactionManager annotationDrivenTransactionManager() {
            return reactiveTransactionManager;
        }
    };
}
于 2020-05-17T17:59:17.037 回答