Spring Boot整合持久层之JPA多数据源
目录
- JPA 多数据源
- 准备工作
- 创建 JPA 配置
- 创建 Repository
- 创建 Controller
JPA 多数据源
JPA 和 MyBatis 配置多数据源类似,不同的是,JPA 配置时主要提供不同的 LocalContainerEntityManagerFactoryBean 以及事务管理器。
项目目录结构如下:
准备工作
创建 Spring Boot Web 项目,添加如下依赖:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-jpa</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid-spring-boot-starter</artifactId> <version>1.1.10</version> </dependency> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <scope>runtime</scope> </dependency>
application.properties 配置
注意:这里的配置与配置单独的 JPA 有区别,因为在后文的配置中要从 JpaProperties 中的 getProperties 方法中获取所有的 JPA 相关配置,因此这里的属性前缀都是 spring.jpa.properties。
server.port=8081
# 数据源1
spring.datasource.one.type=com.alibaba.druid.pool.DruidDataSource
spring.datasource.one.url=jdbc:mysql://localhost:3306/chapter05-1
spring.datasource.one.username=root
spring.datasource.one.password=root
# 数据源2
spring.datasource.two.type=com.alibaba.druid.pool.DruidDataSource
spring.datasource.two.url=jdbc:mysql://localhost:3306/chapter05-2
spring.datasource.two.username=root
spring.datasource.two.password=root
# JPA 配置
spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.MySQL57InnoDBDialect
spring.jpa.properties.database=mysql
spring.jpa.properties.hibernate.hbm2ddl.auto=update
spring.jpa.properties.show-sql= true
数据源配置 DataSourceConfig
@Configuration public class DataSourceConfig { @Bean @ConfigurationProperties("spring.datasource.one") @Primary DataSource dsOne() { return DruidDataSourceBuilder.create().build(); } @Bean @ConfigurationProperties("spring.datasource.two") DataSource dsTwo() { return DruidDataSourceBuilder.create().build(); } }
实体类 User
@Entity(name = "t_user") public class User { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; private String name; private String gender; private Integer age; @Override public String toString() { return "User{" + "id=" + id + ", name='" + name + '\'' + ", gender='" + gender + '\'' + ", age=" + age + '}'; } public Integer getId() { return id; } public void setId(Integer id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getGender() { return gender; } public void setGender(String gender) { this.gender = gender; } public Integer getAge() { return age; } public void setAge(Integer age) { this.age = age; } }
创建 JPA 配置
接下来是核心配置,根据两个配置好的数据源创建两个不同的 JPA 配置
@Configuration @EnableTransactionManagement @EnableJpaRepositories(basePackages = "com.sang.dao1", entityManagerFactoryRef = "entityManagerFactoryBeanOne", transactionManagerRef = "platformTransactionManagerOne") public class JpaConfigOne { @Resource(name = "dsOne") DataSource dsOne; @Autowired JpaProperties jpaProperties; @Bean @Primary LocalContainerEntityManagerFactoryBean entityManagerFactoryBeanOne( EntityManagerFactoryBuilder builder) { return builder.dataSource(dsOne) .properties(jpaProperties.getProperties()) .packages("com.sang.model") .persistenceUnit("pu1") .build(); } @Bean PlatformTransactionManager platformTransactionManagerOne( EntityManagerFactoryBuilder builder) { LocalContainerEntityManagerFactoryBean factoryOne = entityManagerFactoryBeanOne(builder); return new JpaTransactionManager(factoryOne.getObject()); } }
代码解释:
- 使用 @EnableJpaRepositories 注解进行 JPA 配置,该注解中主要配置三个属性:basePackages、entityManagerFactoryRef、transactionManagerRef。其中 basePackages 用来指定 Repository 所在的位置,entityManagerFactoryRef 用来指定实体类管理工厂 Bean 的名称,transactionManagerRef 用来指定事务管理器的引用名称,这里的引用名称就是 JpaConfigOne 类中注册的 Bean 的名称(默认的 Bean 名称为方法名)
- 创建 LocalContainerEntityManagerFactoryBean ,该 Bean 将用来提供 EntityManager 实例,在该类的创建过程中,首先配置数据源,然后设置 JPA 相关配置(JpaProperties 由系统自动加载),再设置实体类所在的位置,最后配置持久化单位名,若项目中只有一个 EntityManagerFactory,则 persistenceUnit 可以省略掉,若有多个,则必须明确指定持久化单元名
- 由于项目中会提供两个 LocalContainerEntityManagerFactoryBean 实例,@Primary 注解表示当存在多个 LocalContainerEntityManagerFactoryBean 实例时,该实例将被优先使用
- platformTransactionManagerOne() 方法表示创建一个事务管理器。JpaTransactionManager 提供对单个 EntityManagerFactory 的事务支持,专门用于解决 JPA 中的事务管理
以下是第二个 JPA 配置
注意:LocalContainerEntityManagerFactoryBean 实例不需要添加 @Primary 注解
@Configuration @EnableTransactionManagement @EnableJpaRepositories(basePackages = "com.sang.dao2", entityManagerFactoryRef = "entityManagerFactoryBeanTwo", transactionManagerRef = "platformTransactionManagerTwo") public class JpaConfigTwo { @Resource(name = "dsTwo") DataSource dsTwo; @Autowired JpaProperties jpaProperties; @Bean LocalContainerEntityManagerFactoryBean entityManagerFactoryBeanTwo( EntityManagerFactoryBuilder builder) { return builder.dataSource(dsTwo) .properties(jpaProperties.getProperties()) .packages("com.sang.model") .persistenceUnit("pu2") .build(); } @Bean PlatformTransactionManager platformTransactionManagerTwo( EntityManagerFactoryBuilder builder) { LocalContainerEntityManagerFactoryBean factoryTwo = entityManagerFactoryBeanTwo(builder); return new JpaTransactionManager(factoryTwo.getObject()); } }
创建 Repository
分别在 com.sang.dao1 和 com.sang.dao2 包下创建两个 Repository
UserDao
public interface UserDao extends JpaRepository<User,Integer> { }
UserDao2
public interface UserDao2 extends JpaRepository<User,Integer> { }
创建 Controller
简便起见,这里省略掉 Service 层,将 UserDao 直接注入 Controller 中
@RestController public class UserController { @Autowired UserDao userDao; @Autowired UserDao2 userDao2; @GetMapping("/test1") public void test1() { User u1 = new User(); u1.setAge(55); u1.setName("鲁迅"); u1.setGender("男"); userDao.save(u1); User u2 = new User(); u2.setAge(80); u2.setName("泰戈尔"); u2.setGender("男"); userDao2.save(u2); } }
http://localhost:8081/test1,查看数据库,即可看到数据库中的表和数据都已经存在了,如下
到此这篇关于Spring Boot整合持久层之JPA多数据源的文章就介绍到这了,更多相关Spring Boot JPA多数据源内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!