0

0

Spring Boot多数据源JPA配置与原生查询实践

花韻仙語

花韻仙語

发布时间:2025-08-05 15:14:01

|

366人浏览过

|

来源于php中文网

原创

Spring Boot多数据源JPA配置与原生查询实践

本文详细介绍了在Spring Boot应用中配置和管理多个JPA数据源的方法,特别聚焦于如何解决在使用原生SQL查询时可能遇到的“表不存在”问题。通过引入PersistenceUnitName明确绑定EntityManager到特定数据源,并结合@EnableJpaRepositories和@Transactional注解,确保不同数据源上的实体管理和事务操作的正确性与隔离性。

1. 引言与问题背景

在企业级应用开发中,为了满足不同的业务需求或数据隔离策略,一个spring boot应用常常需要连接到多个数据库。spring data jpa为多数据源配置提供了强大的支持。然而,当涉及到原生sql查询时,开发者可能会遇到一个常见的问题:尽管非原生jpa查询(如通过repository方法)在特定数据源上运行正常,但原生sql查询却错误地指向了默认或主数据库,导致“表不存在”(relation "table_name" does not exist)的错误。

这个问题通常源于EntityManager未能正确地与预期的特定数据源关联。Spring JPA在默认情况下,如果没有明确指定,可能会将@PersistenceContext注入的EntityManager与主数据源关联。解决此问题的关键在于明确地将每个数据源的EntityManager绑定到一个唯一的持久化单元名称(PersistenceUnitName)。

2. 多数据源配置核心:PersistenceUnitName

为了确保EntityManager能够正确地指向其所属的数据源,我们需要在配置LocalContainerEntityManagerFactoryBean时为其指定一个唯一的PersistenceUnitName,并在需要注入EntityManager的地方通过@PersistenceContext注解引用这个名称。

2.1 数据源与JPA配置类

以下是一个针对名为“pims”的次要数据源的配置示例,展示了如何设置PersistenceUnitName:

package com.xxxx.powwow.config;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;
import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
import org.springframework.orm.jpa.JpaTransactionManager;
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean;
import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter;
import org.springframework.transaction.PlatformTransactionManager;
import javax.sql.DataSource;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.HashMap;

@Configuration
@PropertySource({"classpath:application.properties"})
@EnableJpaRepositories(
        basePackages = {"com.xxxx.powwow.dao.pims", "com.xxxx.powwow.repositories.pims"},
        entityManagerFactoryRef = "pimsEntityManager", // 引用pims数据源的EntityManagerFactory
        transactionManagerRef = "pimsTransactionManager") // 引用pims数据源的事务管理器
public class PersistencePimsAutoConfiguration {

    private Logger logger = LogManager.getLogger(PersistencePimsAutoConfiguration.class);

    @Value("${spring.datasource1.jdbc-url}")
    private String url;
    @Value("${spring.datasource1.username}")
    private String username;
    @Value("${spring.jpa.hibernate.ddl-auto}")
    private String hbm2ddl;
    @Value("${spring.jpa.database-platform}")
    private String platform;
    @Value("${spring.jpa.properties.hibernate.dialect}")
    private String dialect;
    @Value("${spring.profiles.active}")
    private String profile;

    /**
     * 配置pims数据源。
     * 使用@ConfigurationProperties注解从application.properties中读取以"spring.datasource1"为前缀的配置。
     */
    @Bean
    @ConfigurationProperties(prefix="spring.datasource1")
    public DataSource pimsDataSource() {
        return DataSourceBuilder.create().build();
    }

    /**
     * 配置pims数据源的EntityManagerFactory。
     * 关键在于通过em.setPersistenceUnitName("pimsPersistenceUnit")设置唯一的持久化单元名称。
     */
    @Bean(name = "pimsEntityManager") // 明确指定Bean名称
    public LocalContainerEntityManagerFactoryBean pimsEntityManager() {
        LocalContainerEntityManagerFactoryBean em = new LocalContainerEntityManagerFactoryBean();
        em.setDataSource(pimsDataSource());
        em.setPackagesToScan(new String[] {"com.xxxx.powwow.entities.pims"}); // 扫描pims数据源对应的实体类包
        em.setPersistenceUnitName("pimsPersistenceUnit"); // 设定唯一的持久化单元名称

        HibernateJpaVendorAdapter vendorAdapter = new HibernateJpaVendorAdapter();
        em.setJpaVendorAdapter(vendorAdapter);

        HashMap properties = new HashMap<>();
        properties.put("hibernate.hbm2ddl.auto", hbm2ddl);
        properties.put("hibernate.dialect", dialect);
        em.setJpaPropertyMap(properties);

        String host = null;
        try {
            host = InetAddress.getLocalHost().getHostName();
        } catch (UnknownHostException e) {
            throw new RuntimeException(e);
        }
        logger.info("Setting spring.datasource1 (pims): hibernate.hbm2ddl.auto='"+hbm2ddl+"', platform='"+platform+"', url='"+url+"', username='"+username+"', host='"+host+"', profile='"+profile+"'.");
        return em;
    }

    /**
     * 配置pims数据源的事务管理器。
     */
    @Bean(name = "pimsTransactionManager") // 明确指定Bean名称
    public PlatformTransactionManager pimsTransactionManager() {
        JpaTransactionManager transactionManager = new JpaTransactionManager();
        transactionManager.setEntityManagerFactory(pimsEntityManager().getObject());
        return transactionManager;
    }
}

关键点说明:

  • @Configuration:声明这是一个配置类。
  • @PropertySource({"classpath:application.properties"}):加载外部属性文件。
  • @EnableJpaRepositories:启用JPA仓库功能。
    • basePackages:指定该数据源对应的Repository和DAO接口所在的包。
    • entityManagerFactoryRef:引用该数据源的EntityManagerFactory Bean名称。
    • transactionManagerRef:引用该数据源的PlatformTransactionManager Bean名称。
  • pimsDataSource():通过DataSourceBuilder和@ConfigurationProperties构建数据源,从application.properties中读取spring.datasource1前缀的配置。
  • pimsEntityManager():
    • em.setDataSource(pimsDataSource()):关联数据源。
    • em.setPackagesToScan():指定该数据源对应的实体类所在的包,Hibernate会扫描这些包以发现@Entity注解的类。
    • em.setPersistenceUnitName("pimsPersistenceUnit"):这是解决问题的核心。它为该EntityManagerFactory定义了一个唯一的逻辑名称,后续在注入EntityManager时可以通过此名称进行引用。
    • em.setJpaPropertyMap():设置Hibernate相关的JPA属性。
  • pimsTransactionManager():配置事务管理器,并将其与pimsEntityManager关联。

2.2 DAO层使用EntityManager

在DAO(数据访问对象)层,当需要使用EntityManager执行原生查询时,必须明确指定它应该关联哪个持久化单元。

package com.xxxx.powwow.dao.pims;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import javax.persistence.Query;
import java.util.Date;
import java.util.List;

@Component
@Transactional("pimsTransactionManager") // 确保使用pims数据源的事务管理器
public class BookingHistoryReportDao {

    private Logger logger = LogManager.getLogger(BookingHistoryReportDao.class);

    // 通过unitName明确指定注入pims数据源的EntityManager
    @PersistenceContext(unitName = "pimsPersistenceUnit") 
    private EntityManager entityManager;

    public void executeBookingHistoryReport(Date startDate, Date endDate, List companyIds) {
        final String sql = getSQLBookingHistoryReportDao();
        try {
            // 使用正确关联的EntityManager执行原生查询
            Query qry = entityManager.createNativeQuery(sql);
            List merchants = qry.getResultList();
            logger.info("Done executing native query on pims database.");
        } catch (Exception e) {
            logger.error("Error executing query for BookingHistoryReport.", e);
            logger.info("SQL attempted: " + sql);
        }
    }

    private String getSQLBookingHistoryReportDao() {
        // 假设Merchants表在pims数据库中
        return "select company_name from Merchants limit 100";
    }
}

关键点说明:

  • @Component:声明这是一个Spring组件。
  • @Transactional("pimsTransactionManager"):指定该DAO类中的方法将使用名为pimsTransactionManager的事务管理器。这对于确保操作在正确的数据库事务上下文中执行至关重要。
  • @PersistenceContext(unitName = "pimsPersistenceUnit"):这是解决问题的另一个核心部分。它告诉Spring,注入的EntityManager实例必须是与pimsPersistenceUnit这个持久化单元关联的那个。这样,当entityManager.createNativeQuery()被调用时,它就会在pims数据库上执行查询,而不是默认或主数据库。

3. 注意事项与最佳实践

  1. 命名规范: 为PersistenceUnitName、EntityManagerFactory和TransactionManager使用清晰且具有辨识度的名称,以便于管理和维护。

    采风问卷
    采风问卷

    采风问卷是一款全新体验的调查问卷、表单、投票、评测的调研平台,新奇的交互形式,漂亮的作品,让客户眼前一亮,让创作者获得更多的回复。

    下载
  2. 包扫描: 确保@EnableJpaRepositories的basePackages和LocalContainerEntityManagerFactoryBean的setPackagesToScan正确地指向了各自数据源的Repository/DAO接口和实体类。错误的包扫描可能导致实体无法被识别或Repository无法被创建。

  3. 事务管理: 在使用多数据源时,务必通过@Transactional(value = "yourTransactionManagerRef")明确指定事务管理器。否则,Spring可能会使用默认的事务管理器,导致事务操作在错误的数据库上执行。

  4. 属性配置: application.properties或application.yml中应包含所有数据源的详细配置,例如:

    # Primary DataSource (e.g., datasource0)
    spring.datasource.url=jdbc:postgresql://localhost:5432/powwow
    spring.datasource.username=user_powwow
    spring.datasource.password=password_powwow
    spring.datasource.driver-class-name=org.postgresql.Driver
    
    # Secondary DataSource (e.g., datasource1)
    spring.datasource1.jdbc-url=jdbc:postgresql://localhost:5432/pims
    spring.datasource1.username=user_pims
    spring.datasource1.password=password_pims
    spring.datasource1.driver-class-name=org.postgresql.Driver
    
    # JPA/Hibernate properties (can be shared or specific)
    spring.jpa.hibernate.ddl-auto=none
    spring.jpa.database-platform=org.hibernate.dialect.PostgreSQLDialect
    spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.PostgreSQLDialect
  5. 主数据源配置: 如果存在一个主数据源,其配置通常不需要显式指定@Primary或额外的PersistenceUnitName,但为了保持一致性和明确性,也可以为其配置一个PersistenceUnitName。

4. 总结

在Spring Boot应用中处理多JPA数据源,特别是涉及原生SQL查询时,正确配置EntityManager与特定持久化单元的关联至关重要。通过在LocalContainerEntityManagerFactoryBean中设置PersistenceUnitName,并在@PersistenceContext注解中引用该名称,可以确保EntityManager实例正确地指向所需的数据源,从而避免“表不存在”等常见问题,实现多数据源环境下JPA操作的精确控制和隔离。遵循本文提供的配置和使用模式,将有助于构建更健壮、可维护的多数据库Spring Boot应用。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

676

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

320

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

346

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1094

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

357

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

675

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

571

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

414

2024.04.29

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

10

2026.01.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
RunnerGo从入门到精通
RunnerGo从入门到精通

共22课时 | 1.7万人学习

尚学堂Mahout视频教程
尚学堂Mahout视频教程

共18课时 | 3.2万人学习

Linux优化视频教程
Linux优化视频教程

共14课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号