百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

Spring JPA – Multiple Databases

liuian 2025-06-28 15:17 5 浏览

1. 概述

在本教程中,我们将为具有多个数据库的 Spring 数据 JPA 系统实现一个简单的 Spring 配置。

2. 实体

首先,让我们创建两个简单的实体,每个实体都位于一个单独的数据库中。

下面是第一个用户实体:

package com.baeldung.multipledb.model.user;

@Entity
@Table(schema = "users")
public class User {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private int id;

    private String name;

    @Column(unique = true, nullable = false)
    private String email;

    private int age;
}

这是第二个实体,产品

package com.baeldung.multipledb.model.product;

@Entity
@Table(schema = "products")
public class Product {

    @Id
    private int id;

    private String name;

    private double price;
}

我们可以看到,这两个实体也被放置在独立的包中。当我们进入配置时,这将很重要。

3. JPA 存储库

接下来,让我们看一下我们的两个 JPA 存储库,UserRepository

package com.baeldung.multipledb.dao.user;

public interface UserRepository
  extends JpaRepository<User, Integer> { }

产品存储库

package com.baeldung.multipledb.dao.product;

public interface ProductRepository
  extends JpaRepository<Product, Integer> { }

再次注意我们如何在不同的包中创建这两个存储库。

4. 使用 Java 配置 JPA

现在我们将进入实际的 Spring 配置。我们将首先设置两个配置类 — 一个用于用户,另一个用于产品

在每个配置类中,我们需要为 User 定义以下接口:

  • 数据源
  • EntityManagerFactory (userEntityManager )
  • TransactionManager (userTransactionManager)

让我们先看一下用户配置:

@Configuration
@PropertySource({ "classpath:persistence-multiple-db.properties" })
@EnableJpaRepositories(
    basePackages = "com.baeldung.multipledb.dao.user", 
    entityManagerFactoryRef = "userEntityManager", 
    transactionManagerRef = "userTransactionManager"
)
public class PersistenceUserConfiguration {
    @Autowired
    private Environment env;
    
    @Bean
    @Primary
    public LocalContainerEntityManagerFactoryBean userEntityManager() {
        LocalContainerEntityManagerFactoryBean em
          = new LocalContainerEntityManagerFactoryBean();
        em.setDataSource(userDataSource());
        em.setPackagesToScan(
          new String[] { "com.baeldung.multipledb.model.user" });

        HibernateJpaVendorAdapter vendorAdapter
          = new HibernateJpaVendorAdapter();
        em.setJpaVendorAdapter(vendorAdapter);
        HashMap<String, Object> properties = new HashMap<>();
        properties.put("hibernate.hbm2ddl.auto",
          env.getProperty("hibernate.hbm2ddl.auto"));
        properties.put("hibernate.dialect",
          env.getProperty("hibernate.dialect"));
        em.setJpaPropertyMap(properties);

        return em;
    }

    @Primary
    @Bean
    public DataSource userDataSource() {
 
        DriverManagerDataSource dataSource
          = new DriverManagerDataSource();
        dataSource.setDriverClassName(
          env.getProperty("jdbc.driverClassName"));
        dataSource.setUrl(env.getProperty("user.jdbc.url"));
        dataSource.setUsername(env.getProperty("jdbc.user"));
        dataSource.setPassword(env.getProperty("jdbc.pass"));

        return dataSource;
    }

    @Primary
    @Bean
    public PlatformTransactionManager userTransactionManager() {
 
        JpaTransactionManager transactionManager
          = new JpaTransactionManager();
        transactionManager.setEntityManagerFactory(
          userEntityManager().getObject());
        return transactionManager;
    }
}

请注意我们如何通过使用 @Primary 注释 bean 定义来使用 userTransactionManager 作为我们的主事务管理器。每当我们要隐式或显式注入事务管理器而不按名称指定哪个时,这很有帮助。

接下来,让我们讨论


PersistenceProductConfiguration
,其中我们定义了类似的 bean:

@Configuration
@PropertySource({ "classpath:persistence-multiple-db.properties" })
@EnableJpaRepositories(
    basePackages = "com.baeldung.multipledb.dao.product", 
    entityManagerFactoryRef = "productEntityManager", 
    transactionManagerRef = "productTransactionManager"
)
public class PersistenceProductConfiguration {
    @Autowired
    private Environment env;

    @Bean
    public LocalContainerEntityManagerFactoryBean productEntityManager() {
        LocalContainerEntityManagerFactoryBean em
          = new LocalContainerEntityManagerFactoryBean();
        em.setDataSource(productDataSource());
        em.setPackagesToScan(
          new String[] { "com.baeldung.multipledb.model.product" });

        HibernateJpaVendorAdapter vendorAdapter = new HibernateJpaVendorAdapter();
        em.setJpaVendorAdapter(vendorAdapter);
        HashMap<String, Object> properties = new HashMap<>();
        properties.put("hibernate.hbm2ddl.auto",
          env.getProperty("hibernate.hbm2ddl.auto"));
        properties.put("hibernate.dialect",
          env.getProperty("hibernate.dialect"));
        em.setJpaPropertyMap(properties);

        return em;
    }

    @Bean
    public DataSource productDataSource() {
 
        DriverManagerDataSource dataSource
          = new DriverManagerDataSource();
        dataSource.setDriverClassName(
          env.getProperty("jdbc.driverClassName"));
        dataSource.setUrl(env.getProperty("product.jdbc.url"));
        dataSource.setUsername(env.getProperty("jdbc.user"));
        dataSource.setPassword(env.getProperty("jdbc.pass"));

        return dataSource;
    }

    @Bean
    public PlatformTransactionManager productTransactionManager() {
 
        JpaTransactionManager transactionManager
          = new JpaTransactionManager();
        transactionManager.setEntityManagerFactory(
          productEntityManager().getObject());
        return transactionManager;
    }
}

5. 简单测试

最后,让我们测试一下我们的配置。

为此,我们将创建每个实体的实例并确保已创建它:

@RunWith(SpringRunner.class)
@SpringBootTest
@EnableTransactionManagement
public class JpaMultipleDBIntegrationTest {
 
    @Autowired
    private UserRepository userRepository;

    @Autowired
    private ProductRepository productRepository;

    @Test
    @Transactional("userTransactionManager")
    public void whenCreatingUser_thenCreated() {
        User user = new User();
        user.setName("John");
        user.setEmail("john@test.com");
        user.setAge(20);
        user = userRepository.save(user);

        assertNotNull(userRepository.findOne(user.getId()));
    }

    @Test
    @Transactional("userTransactionManager")
    public void whenCreatingUsersWithSameEmail_thenRollback() {
        User user1 = new User();
        user1.setName("John");
        user1.setEmail("john@test.com");
        user1.setAge(20);
        user1 = userRepository.save(user1);
        assertNotNull(userRepository.findOne(user1.getId()));

        User user2 = new User();
        user2.setName("Tom");
        user2.setEmail("john@test.com");
        user2.setAge(10);
        try {
            user2 = userRepository.save(user2);
        } catch (DataIntegrityViolationException e) {
        }

        assertNull(userRepository.findOne(user2.getId()));
    }

    @Test
    @Transactional("productTransactionManager")
    public void whenCreatingProduct_thenCreated() {
        Product product = new Product();
        product.setName("Book");
        product.setId(2);
        product.setPrice(20);
        product = productRepository.save(product);

        assertNotNull(productRepository.findOne(product.getId()));
    }
}

6. Spring 启动中的多个数据库

Spring Boot 可以简化上述配置。

默认情况下,Spring Boot 将使用 spring.datasource.* 前缀的配置属性实例化其默认数据源

spring.datasource.jdbcUrl = [url]
spring.datasource.username = [username]
spring.datasource.password = [password]

我们现在希望继续使用相同的方法来配置第二个数据源,但使用不同的属性命名空间

spring.second-datasource.jdbcUrl = [url]
spring.second-datasource.username = [username]
spring.second-datasource.password = [password]

因为我们希望 Spring Boot 自动配置来获取这些不同的属性(并实例化两个不同的数据源),所以我们将定义两个类似于前面部分的配置类:

@Configuration
@PropertySource({"classpath:persistence-multiple-db-boot.properties"})
@EnableJpaRepositories(
  basePackages = "com.baeldung.multipledb.dao.user",
  entityManagerFactoryRef = "userEntityManager",
  transactionManagerRef = "userTransactionManager")
public class PersistenceUserAutoConfiguration {
    
    @Primary
    @Bean
    @ConfigurationProperties(prefix="spring.datasource")
    public DataSource userDataSource() {
        return DataSourceBuilder.create().build();
    }
    // userEntityManager bean 

    // userTransactionManager bean
}
@Configuration
@PropertySource({"classpath:persistence-multiple-db-boot.properties"})
@EnableJpaRepositories(
  basePackages = "com.baeldung.multipledb.dao.product", 
  entityManagerFactoryRef = "productEntityManager", 
  transactionManagerRef = "productTransactionManager")
public class PersistenceProductAutoConfiguration {
   
    @Bean
    @ConfigurationProperties(prefix="spring.second-datasource")
    public DataSource productDataSource() {
        return DataSourceBuilder.create().build();
    }
   
    // productEntityManager bean 

    // productTransactionManager bean
}

现在,我们已经根据 Boot 自动配置约定在


persistence-multiple-db-boot.properties
中定义了数据源属性。

有趣的部分是用@ConfigurationProperties注释数据源 Bean 创建方法。我们只需要指定相应的配置前缀在这种方法中,我们使用的是DataSourceBuilder,Spring Boot将自动处理其余的工作。

但是,如何将配置的属性注入到数据源配置中呢?

DataSourceBuilder 上调用 build() 方法时,它将调用其私有 bind() 方法:

public T build() {
    Class<? extends DataSource> type = getType();
    DataSource result = BeanUtils.instantiateClass(type);
    maybeGetDriverClassName();
    bind(result);
    return (T) result;
}

此私有方法执行大部分自动配置魔术,将解析的配置绑定到实际的数据源实例:

private void bind(DataSource result) {
    ConfigurationPropertySource source = new MapConfigurationPropertySource(this.properties);
    ConfigurationPropertyNameAliases aliases = new ConfigurationPropertyNameAliases();
    aliases.addAliases("url", "jdbc-url");
    aliases.addAliases("username", "user");
    Binder binder = new Binder(source.withAliases(aliases));
    binder.bind(ConfigurationPropertyName.EMPTY, Bindable.ofInstance(result));
}

虽然我们自己不必接触任何这些代码,但了解 Spring Boot 自动配置的幕后发生了什么仍然很有用。

除此之外,事务管理器和实体管理器 bean 配置与标准 Spring 应用程序相同。

7. 结论

本文是对如何配置我们的 Spring Data JPA 项目以使用多个数据库的实际概述。

完整的实现可以在 GitHub 项目中找到。这是一个基于 Maven 的项目,因此应该易于导入和按原样运行。

参考 :


https://www.baeldung.com/spring-data-jpa-multiple-databases

github:https://github.com/eugenp/tutorials/tree/master/persistence-modules/spring-data-jpa-enterprise-2

相关推荐

什么是Java消息系统(JMS)?(java消息机制原理)

在JavaEE应用程序的典型体系结构中,数据访问层具有与数据存储进行通信的技术,如Java持久性API(JPA)、Java连接器体系结构(JCA),它将应用程序连接到企业信息系统(EIS);和Jav...

学习java没规划?2020最新路线图,大堆资源秒变大神

为什么编程就跟数学一样弯下头捡铅笔从此就听不懂?为什么学完之后感觉简单,刷题却像刷人生?为什么这个规划图这么受欢迎,这么多人看?为什么这个小图能在我们官网占据这么重要的位置?请观看本文《黑马程序员——...

Spring Data JPA:用最少的代码搞定数据库操作

SpringDataJPA:用最少的代码搞定数据库操作今天咱们聊聊SpringDataJPA这个神器,它简直就是JAVA开发者的好帮手,特别是那些被复杂的SQL语句折磨得苦不堪言的朋友。想象一...

Spring Data JPA的高级查询:从基础到进阶

SpringDataJPA的高级查询:从基础到进阶在Java的生态系统中,SpringDataJPA以其强大的功能和便捷性,成为了许多开发者处理数据库操作的首选工具。它简化了JPA的操作流程,...

客观对比Spring boot JPA 和Mybatis plus

SpringBootJPA和MyBatisPlus都是在Java生态系统中广泛使用的持久化框架,用于简化和加速与关系型数据库的交互。然而,它们在设计哲学、用法和功能上有一些不同。设计哲...

Hibernate 和 JPA(hibernate和jpa的区别)

Hibernate在5以后的版本中全面推进使用JPA的查询语法,甚至准备废弃掉自己已有的查询语法。什么是JPAJPA全称为JavaPersistenceAPI,Java持久化A...

[Java速成] 数据库基础,Connector/J、JDBC、JPA的关系(day 7)

先来看一个问题:在创建Java项目,选择数据库依赖时,如下图,你知道该如何正确的选择合适你项目的依赖吗?看完这篇文章,就能有个很好的认知了。后续的文章中,我们也需要使用数据库,所以在正式开始使用...

Java从入门到实战总踩坑?这6大核心+实战案例,帮你打通任督二脉

开篇:你是不是也这样?学Java越学越懵!最近有个刚学Java的朋友跟我吐槽:“看了3个月教程,语法背得滚瓜烂熟,结果写个‘用户登录’的小程序都卡壳——对象不会用、内存总溢出、多线程代码越调越乱……”...

盘点JPA中的骚操作(jpa使用教程)

相信大多数的Java后端开发者都听说过MyBatis,但是用过JPA的人却相对较少,有些人可能没有听说过。JPA在国外比较火用的人也比较多,国内大多被MyBatis的简单易上手所吸引。相对的,JPA和...

SpringBoot中使用Spring Data JPA

Springboot中如何集成springdatajpa一什么是ORM?ORM即Object-RelationlMapping,它的作用是在关系型数据库和对象之间作一个映射,这样,我们在具体...

SpringBoot整合Spring Data JPA(SpringBoot整合JWT)

本篇要点简单介绍JPA。介绍快速SpringBoot快速整合JPAJPA是啥?TheJavaPersistenceAPIisastandardtechnologythatletsy...

Spring Boot中使用JPA与MyBatis有什么区别?

JPA(JavaPersistenceAPI)和MyBatis都是在SpringBoot应用程序中比较常用的两种持久化框架,他们各自有各自的优点和缺点,以及适合的使用场景,下面我们就来详细介绍...

Spring JPA – Multiple Databases

1.概述在本教程中,我们将为具有多个数据库的Spring数据JPA系统实现一个简单的Spring配置。2.实体首先,让我们创建两个简单的实体,每个实体都位于一个单独的数据库中。下面是第...

Spring Data JPA 的 JpaRepository与MyBatis区别对比

SpringDataJPA的JpaRepository和MyBatis是Java生态中两种非常流行但设计理念和使用方式截然不同的持久层框架。它们的核心区别在于如何管理对象-关系映射(...

Spring Data Jpa 介绍和详细入门案例搭建

1.SpringDataJPA的概念在介绍SpringDataJPA的时候,我们首先认识下Hibernate。Hibernate是数据访问解决技术的绝对霸主,使用O/R映射(Object-Re...