Spring Boot 配置和使用多个数据源

1. 概述

Spring Boot 应用程序的典型方案是将数据存储在单个关系数据库中。但我们有时需要访问多个数据库。

在本教程中,我们将学习如何通过弹簧启动配置和使用多个数据源。

要了解如何处理单个数据源,请查看我们对Spring数据JPA的介绍

2. 默认行为

让我们记住在弹簧启动中声明数据源在应用程序.yml中的样子:

spring:
  datasource:
    url: ...
    username: ...
    password: ...
    driverClassname: ...

在内部,Spring 将这些设置映射到个实例。

让我们来看看实现:

@ConfigurationProperties(prefix = "spring.datasource")
public class DataSourceProperties implements BeanClassLoaderAware, InitializingBean {

    // ...

    /**
     * Fully qualified name of the JDBC driver. Auto-detected based on the URL by default.
     */
    private String driverClassName;

    /**
     * JDBC URL of the database.
     */
    private String url;

    /**
     * Login username of the database.
     */
    private String username;

    /**
     * Login password of the database.
     */
    private String password;

    // ...

}

我们应该指出@ConfigurationProperties注释,它将配置的属性自动映射到Java对象。

3. 扩展默认值

因此,要使用多个数据源,我们需要在Spring的应用程序上下文中声明具有不同映射的多个bean。

我们可以通过使用配置类来做到这一点:

@Configuration
public class TodoDatasourceConfiguration {

    @Bean
    @ConfigurationProperties("spring.datasource.todos")
    public DataSourceProperties todosDataSourceProperties() {
        return new DataSourceProperties();
    }

    @Bean
    @ConfigurationProperties("spring.datasource.topics")
    public DataSourceProperties topicsDataSourceProperties() {
        return new DataSourceProperties();
    }

}

数据源的配置必须如下所示:

spring:
  datasource:
    todos:
      url: ...
      username: ...
      password: ...
      driverClassName: ...
    topics:
      url: ...
      username: ...
      password: ...
      driverClassName: ...

然后,我们可以使用数据源属性对象创建数据源:

@Bean
public DataSource todosDataSource() {
    return todosDataSourceProperties()
      .initializeDataSourceBuilder()
      .build();
}

@Bean
public DataSource topicsDataSource() {
    return topicsDataSourceProperties()
      .initializeDataSourceBuilder()
      .build();
}

4. 春季数据 JDBC

使用春季数据 JDBC 时,我们还需要为每个数据源配置一个 Jdbc 模板实例:

@Bean
public JdbcTemplate todosJdbcTemplate(@Qualifier("todosDataSource") DataSource dataSource) {
    return new JdbcTemplate(dataSource);
}

@Bean
public JdbcTemplate topicsJdbcTemplate(@Qualifier("topicsDataSource") DataSource dataSource) {
    return new JdbcTemplate(dataSource);
}

然后,我们也可以通过指定@Qualifier来使用它们:

@Autowired
@Qualifier("topicsJdbcTemplate")
JdbcTemplate jdbcTemplate;

5. 春季数据JPA

使用春季数据JPA时,我们希望使用如下所示的存储库,其中Todo是实体:

public interface TodoRepository extends JpaRepository<Todo, Long> {}

因此,我们需要为每个数据源声明实体管理器工厂:

@Configuration
@EnableTransactionManagement
@EnableJpaRepositories(
  basePackageClasses = Todo.class,
  entityManagerFactoryRef = "todosEntityManagerFactory",
  transactionManagerRef = "todosTransactionManager"
)
public class TodoJpaConfiguration {

    @Bean
    public LocalContainerEntityManagerFactoryBean todosEntityManagerFactory(
      Qualifier("todosDataSource") DataSource dataSource,
      EntityManagerFactoryBuilder builder) {
        return builder
          .dataSource(todosDataSource())
          .packages(Todo.class)
          .build();
    }

    @Bean
    public PlatformTransactionManager todosTransactionManager(
      @Qualifier("todosEntityManagerFactory") LocalContainerEntityManagerFactoryBean todosEntityManagerFactory) {
        return new JpaTransactionManager(Objects.requireNonNull(todosEntityManagerFactory.getObject()));
    }

}

让我们看一下我们应该注意的一些限制。

我们需要拆分包,以便为每个数据源允许一个@EnableJpaRepositories

不幸的是,要注入实体管理器工厂生成器,我们需要将其中一个数据源声明为@Primary

这是因为实体管理器工厂生成器是在 org.springframework.boot.自动配置.orm.jpa.JpaBaseConfimatic 中声明的,并且此类需要注入单个数据源。通常,框架的某些部分可能不需要配置多个数据源。

6. 配置光连接池

如果要配置 Hikari,只需向数据源定义添加一个@ConfigurationProperties

@Bean
@ConfigurationProperties("spring.datasource.todos.hikari")
public DataSource todosDataSource() {
    return todosDataSourceProperties()
      .initializeDataSourceBuilder()
      .build();
}

然后,我们可以将以下行插入到应用程序.properties文件中:

spring.datasource.todos.hikari.connectionTimeout=30000 
spring.datasource.todos.hikari.idleTimeout=600000 
spring.datasource.todos.hikari.maxLifetime=1800000 

7. 结论

在本文中,我们学习了如何使用弹簧启动配置多个数据源。

我们看到我们需要一些配置,并且在偏离标准时可能存在陷阱,但最终是可能的。

与往常一样,所有代码都可以在 GitHub 上使用

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值