0


Spring Boot 多数据源配置(JPA)

前言

一般一个系统至少有一个数据源,用来持久化业务数据以及查询。单个数据源的系统很常见,在 Spring Boot 框架下配置也很简单。在约定大于配置这个思想下,只需要在配置文件中简单配置下数据库连接信息就行了。

除了单数据源系统,在实际工作中,还会遇到多数据源系统。有可能是因为业务需要,以实际中做过的资产清查相关的项目为例,数据的存储分成了两个库,一个当前库和归档库,系统就需要配置两个数据源来满足业务需求。也有可能是技术架构,比如 MySQL 数据库使用了主从复制架构来实现读写分离,然后在应用层做分流,就需要配置主库和从库数据源,并在代码中决定使用哪个库进行数据库操作。

以下是Spring Boot JPA 多数据源的配置教程

前置环境

JDK8 + SringBoot2 + MySQL8

分别创建数据库 test1 test2

分别在两个数据库中创建 user 表

create table user (
id int auto_increment primary key,
username varchar(255),
password varchar(255)
);

在test1.user 表中插入数据

insert into user(username, password) values('张三', '123456');

在test2.user 表中插入数据

insert into user(username, password) values('李四', '123456');

pom

  1. <dependencyManagement>
  2. <dependencies>
  3. <dependency>
  4. <groupId>org.springframework.boot</groupId>
  5. <artifactId>spring-boot-dependencies</artifactId>
  6. <version>${spring-boot.version}</version>
  7. <type>pom</type>
  8. <scope>import</scope>
  9. </dependency>
  10. </dependencies>
  11. </dependencyManagement>
  12. <dependencies>
  13. <dependency>
  14. <groupId>org.springframework.boot</groupId>
  15. <artifactId>spring-boot-starter-web</artifactId>
  16. </dependency>
  17. <dependency>
  18. <groupId>org.springframework.boot</groupId>
  19. <artifactId>spring-boot-starter-data-jpa</artifactId>
  20. </dependency>
  21. <dependency>
  22. <groupId>mysql</groupId>
  23. <artifactId>mysql-connector-java</artifactId>
  24. </dependency>
  25. <dependencies>

yml

  1. server:
  2. port: 8888
  3. spring:
  4. datasource:
  5. primary:
  6. url: jdbc:mysql://localhost:3306/test1?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=UTC&allowPublicKeyRetrieval=true
  7. username: root
  8. password: mysql
  9. driver-class-name: com.mysql.cj.jdbc.Driver
  10. secondary:
  11. url: jdbc:mysql://localhost:3306/test2?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=UTC&allowPublicKeyRetrieval=true
  12. username: root
  13. password: mysql
  14. driver-class-name: com.mysql.cj.jdbc.Driver
  15. jpa:
  16. primary:
  17. show-sql: true
  18. properties:
  19. hibernate:
  20. hbm2ddl:
  21. auto: update
  22. dialect: org.hibernate.dialect.MySQL5InnoDBDialect
  23. secondary:
  24. show-sql: true
  25. properties:
  26. hibernate:
  27. hbm2ddl:
  28. auto: update
  29. dialect: org.hibernate.dialect.MySQL5InnoDBDialect

Entity

主库和从库的实体类需要放在不同的包下,以本文为例

主库的实体类包路径为:com.jpa.entity.primary

  1. @Entity
  2. @Table ( name = "user")
  3. public class PrimaryUserEntity {
  4. private Integer id;
  5. private String username;
  6. private String password;
  7. @Id
  8. @GeneratedValue ( strategy = GenerationType.IDENTITY)
  9. @Column ( name = "id" )
  10. public Integer getId() {
  11. return id;
  12. }
  13. public void setId(Integer id) {
  14. this.id = id;
  15. }
  16. @Basic
  17. @Column ( name = "username" )
  18. public String getUsername() {
  19. return username;
  20. }
  21. public void setUsername(String username) {
  22. this.username = username;
  23. }
  24. @Basic
  25. @Column ( name = "password" )
  26. public String getPassword() {
  27. return password;
  28. }
  29. public void setPassword(String password) {
  30. this.password = password;
  31. }
  32. }

从库的实体类包路径为:com.jpa.entity.secondary

  1. @Entity
  2. @Table ( name = "user")
  3. public class SecondaryUserEntity {
  4. private Integer id;
  5. private String username;
  6. private String password;
  7. @Id
  8. @GeneratedValue ( strategy = GenerationType.IDENTITY)
  9. @Column ( name = "id" )
  10. public Integer getId() {
  11. return id;
  12. }
  13. public void setId(Integer id) {
  14. this.id = id;
  15. }
  16. @Basic
  17. @Column ( name = "username" )
  18. public String getUsername() {
  19. return username;
  20. }
  21. public void setUsername(String username) {
  22. this.username = username;
  23. }
  24. @Basic
  25. @Column ( name = "password" )
  26. public String getPassword() {
  27. return password;
  28. }
  29. public void setPassword(String password) {
  30. this.password = password;
  31. }
  32. }

Dao

主库和从库的Dao需要放在不同的包下,以本文为例

主库的Dao包路径为:com.jpa.dao.primary

  1. @Repository (value = IPrimaryUserDao.DAO_BEAN_NAME )
  2. public interface IPrimaryUserDao extends JpaRepository<PrimaryUserEntity, Integer> {
  3. String DAO_BEAN_NAME = "primaryUserDao";
  4. }

从库的Dao包路径为:com.jpa.dao.secondary

  1. @Repository (value = ISecondaryUserDao.DAO_BEAN_NAME )
  2. public interface ISecondaryUserDao extends JpaRepository<SecondaryUserEntity, Integer> {
  3. String DAO_BEAN_NAME = "secondaryUserDao";
  4. }

Config

记得将类中的两个包路径修改成自己项目的包路径

REPOSITORY_PACKAGE

ENTITY_PACKAGE

主库数据源配置

  1. @Configuration
  2. @EnableTransactionManagement
  3. @EnableJpaRepositories (
  4. basePackages = PrimaryDatasourceAndJpaConfig.REPOSITORY_PACKAGE,
  5. entityManagerFactoryRef = "primaryEntityManagerFactory",
  6. transactionManagerRef = "primaryTransactionManager"
  7. )
  8. public class PrimaryDatasourceAndJpaConfig {
  9. private static final String REPOSITORY_PACKAGE = "com.jpa.dao.primary";
  10. private static final String ENTITY_PACKAGE = "com.jpa.entity.primary";
  11. //--------------数据源配置-------------------
  12. /**
  13. * 扫描spring.datasource.primary开头的配置信息
  14. *
  15. * @return 数据源配置信息
  16. */
  17. @Primary
  18. @Bean(name = "primaryDataSourceProperties")
  19. @ConfigurationProperties(prefix = "spring.datasource.primary")
  20. public DataSourceProperties dataSourceProperties() {
  21. return new DataSourceProperties();
  22. }
  23. /**
  24. * 取主库数据源对象
  25. *
  26. * @param dataSourceProperties 注入名为primaryDataSourceProperties的bean
  27. * @return 数据源对象
  28. */
  29. @Primary
  30. @Bean(name = "primaryDataSource")
  31. public DataSource dataSource(@Qualifier("primaryDataSourceProperties") DataSourceProperties dataSourceProperties) {
  32. return dataSourceProperties.initializeDataSourceBuilder().build();
  33. }
  34. /**
  35. * 扫描spring.jpa.primary开头的配置信息
  36. *
  37. * @return jpa配置信息
  38. */
  39. @Primary
  40. @Bean (name = "primaryJpaProperties")
  41. @ConfigurationProperties (prefix = "spring.jpa.primary")
  42. public JpaProperties jpaProperties() {
  43. return new JpaProperties();
  44. }
  45. /**
  46. * 获取主库实体管理工厂对象
  47. *
  48. * @param primaryDataSource 注入名为primaryDataSource的数据源
  49. * @param jpaProperties 注入名为primaryJpaProperties的jpa配置信息
  50. * @param builder 注入EntityManagerFactoryBuilder
  51. * @return 实体管理工厂对象
  52. */
  53. @Primary
  54. @Bean(name = "primaryEntityManagerFactory")
  55. public LocalContainerEntityManagerFactoryBean entityManagerFactoryBean(
  56. @Qualifier ("primaryDataSource") DataSource primaryDataSource,
  57. @Qualifier("primaryJpaProperties") JpaProperties jpaProperties,
  58. EntityManagerFactoryBuilder builder
  59. ) {
  60. return builder
  61. // 设置数据源
  62. .dataSource(primaryDataSource)
  63. // 设置jpa配置
  64. .properties(jpaProperties.getProperties())
  65. // 设置实体包名
  66. .packages(ENTITY_PACKAGE)
  67. // 设置持久化单元名,用于@PersistenceContext注解获取EntityManager时指定数据源
  68. .persistenceUnit("primaryPersistenceUnit").build();
  69. }
  70. /**
  71. * 获取实体管理对象
  72. *
  73. * @param factory 注入名为primaryEntityManagerFactory的bean
  74. * @return 实体管理对象
  75. */
  76. @Primary
  77. @Bean(name = "primaryEntityManager")
  78. public EntityManager entityManager(@Qualifier("primaryEntityManagerFactory") EntityManagerFactory factory) {
  79. return factory.createEntityManager();
  80. }
  81. /**
  82. * 获取主库事务管理对象
  83. *
  84. * @param factory 注入名为primaryEntityManagerFactory的bean
  85. * @return 事务管理对象
  86. */
  87. @Primary
  88. @Bean(name = "primaryTransactionManager")
  89. public JpaTransactionManager transactionManager(@Qualifier("primaryEntityManagerFactory") EntityManagerFactory factory) {
  90. return new JpaTransactionManager(factory);
  91. }
  92. }

从库数据源配置

  1. @Configuration
  2. @EnableTransactionManagement
  3. @EnableJpaRepositories(
  4. basePackages = SecondaryDatasourceAndJpaConfig.REPOSITORY_PACKAGE,
  5. entityManagerFactoryRef = "secondaryEntityManagerFactory",
  6. transactionManagerRef = "secondaryTransactionManager"
  7. )
  8. public class SecondaryDatasourceAndJpaConfig {
  9. static final String REPOSITORY_PACKAGE = "com.jpa.dao.secondary";
  10. static final String ENTITY_PACKAGE = "com.jpa.entity.secondary";
  11. //--------------数据源配置-------------------
  12. /**
  13. * 扫描spring.datasource.secondary开头的配置信息
  14. *
  15. * @return 数据源配置信息
  16. */
  17. @Bean(name = "secondaryDataSourceProperties")
  18. @ConfigurationProperties(prefix = "spring.datasource.secondary")
  19. public DataSourceProperties dataSourceProperties() {
  20. return new DataSourceProperties();
  21. }
  22. /**
  23. * 获取次数据源对象
  24. *
  25. * @param dataSourceProperties 注入名为secondaryDataSourceProperties的bean
  26. * @return 数据源对象
  27. */
  28. @Bean("secondaryDataSource")
  29. public DataSource dataSource(@Qualifier("secondaryDataSourceProperties") DataSourceProperties dataSourceProperties) {
  30. return dataSourceProperties.initializeDataSourceBuilder().build();
  31. }
  32. /**
  33. * 扫描spring.jpa.secondary
  34. *
  35. * @return jpa配置信息
  36. */
  37. @Bean(name = "secondaryJpaProperties")
  38. @ConfigurationProperties(prefix = "spring.jpa.secondary")
  39. public JpaProperties jpaProperties() {
  40. return new JpaProperties();
  41. }
  42. /**
  43. * 获取次库实体管理工厂对象
  44. *
  45. * @param secondaryDataSource 注入名为secondaryDataSource的数据源
  46. * @param jpaProperties 注入名为secondaryJpaProperties的jpa配置信息
  47. * @param builder 注入EntityManagerFactoryBuilder
  48. * @return 实体管理工厂对象
  49. */
  50. @Bean(name = "secondaryEntityManagerFactory")
  51. public LocalContainerEntityManagerFactoryBean entityManagerFactory(
  52. @Qualifier("secondaryDataSource") DataSource secondaryDataSource,
  53. @Qualifier("secondaryJpaProperties") JpaProperties jpaProperties,
  54. EntityManagerFactoryBuilder builder
  55. ) {
  56. return builder
  57. // 设置数据源
  58. .dataSource(secondaryDataSource)
  59. // 设置jpa配置
  60. .properties(jpaProperties.getProperties())
  61. // 设置实体包名
  62. .packages(ENTITY_PACKAGE)
  63. // 设置持久化单元名,用于@PersistenceContext注解获取EntityManager时指定数据源
  64. .persistenceUnit("secondaryPersistenceUnit").build();
  65. }
  66. /**
  67. * 获取实体管理对象
  68. *
  69. * @param factory 注入名为secondaryEntityManagerFactory的bean
  70. * @return 实体管理对象
  71. */
  72. @Bean(name = "secondaryEntityManager")
  73. public EntityManager entityManager(@Qualifier("secondaryEntityManagerFactory") EntityManagerFactory factory) {
  74. return factory.createEntityManager();
  75. }
  76. /**
  77. * 获取事务管理对象
  78. *
  79. * @param factory 注入名为secondaryEntityManagerFactory的bean
  80. * @return 事务管理对象
  81. */
  82. @Bean(name = "secondaryTransactionManager")
  83. public JpaTransactionManager transactionManager(@Qualifier("secondaryEntityManagerFactory") EntityManagerFactory factory) {
  84. return new JpaTransactionManager(factory);
  85. }
  86. }

Controller

主库

  1. @RestController
  2. @RequestMapping(value = "/primary")
  3. public class PrimaryJpaController {
  4. @Resource
  5. private IPrimaryUserDao primaryUserDao;
  6. @RequestMapping(value = "/findAll", method = RequestMethod.GET)
  7. public List<PrimaryUserEntity> findAll() {
  8. return primaryUserDao.findAll();
  9. }
  10. }

从库

  1. @RestController
  2. @RequestMapping(value = "/secondary")
  3. public class SecondaryJpaController{
  4. @Resource
  5. private ISecondaryUserDao secondaryUserDao;
  6. @RequestMapping(value = "/findAll", method = RequestMethod.GET)
  7. public List<SecondaryUserEntity> findAll() {
  8. return secondaryUserDao.findAll();
  9. }
  10. }

演示

请求 /primary/findAll

请求 /secondary/findAll

标签: spring boot 后端 java

本文转载自: https://blog.csdn.net/typeracer/article/details/141394243
版权归原作者 编程经验分享 所有, 如有侵权,请联系我们删除。

“Spring Boot 多数据源配置(JPA)”的评论:

还没有评论