R2DBC 仓库
本章重点介绍 R2DBC 仓库支持的特点。这建立在使用 Spring Data 仓库中解释的核心仓库支持之上。在阅读本章之前,您应该对其中解释的基本概念有扎实的理解。
用法
要访问存储在关系数据库中的领域实体,您可以使用我们复杂的仓库支持,这可以显著简化实现。为此,请为您的仓库创建一个接口。考虑以下 Person 类
public class Person {
@Id
private Long id;
private String firstname;
private String lastname;
// … getters and setters omitted
}
以下示例显示了前面 Person 类的仓库接口
public interface PersonRepository extends ReactiveCrudRepository<Person, Long> {
// additional custom query methods go here
}
要配置 R2DBC 仓库,您可以使用 @EnableR2dbcRepositories 注解。如果未配置基本包,则基础设施会扫描注解配置类的包。以下示例显示了如何将 Java 配置用于仓库
@Configuration
@EnableR2dbcRepositories
class ApplicationConfig extends AbstractR2dbcConfiguration {
@Override
public ConnectionFactory connectionFactory() {
return …
}
}
由于我们的领域仓库扩展了 ReactiveCrudRepository,它为您提供了响应式 CRUD 操作来访问实体。在 ReactiveCrudRepository 之上,还有 ReactiveSortingRepository,它添加了类似于 PagingAndSortingRepository 的额外排序功能。使用仓库实例仅仅是将其依赖注入到客户端的问题。因此,您可以使用以下代码检索所有 Person 对象
@ExtendWith(SpringExtension.class)
@ContextConfiguration
class PersonRepositoryTests {
@Autowired
PersonRepository repository;
@Test
void readsAllEntitiesCorrectly() {
repository.findAll()
.as(StepVerifier::create)
.expectNextCount(1)
.verifyComplete();
}
@Test
void readsEntitiesByNameCorrectly() {
repository.findByFirstname("Hello World")
.as(StepVerifier::create)
.expectNextCount(1)
.verifyComplete();
}
}
前面的示例使用 Spring 的单元测试支持创建了一个应用程序上下文,该上下文在测试用例中执行基于注解的依赖注入。在测试方法内部,我们使用仓库查询数据库。我们使用 StepVerifier 作为测试辅助工具来验证我们对结果的期望。
结果映射
返回接口或 DTO 投影的查询方法由实际查询产生的结果支持。接口投影通常首先依赖于将结果映射到领域类型,以考虑潜在的 @Column 类型映射,并且实际的投影代理使用可能部分具体化的实体来公开投影数据。
DTO 投影的结果映射取决于实际的查询类型。派生查询使用领域类型来映射结果,并且 Spring Data 仅从领域类型上可用的属性创建 DTO 实例。不支持在 DTO 中声明领域类型上不可用的属性。
基于字符串的查询采用不同的方法,因为实际查询,特别是字段投影和结果类型声明,是紧密相关的。与 @Query 注解的查询方法一起使用的 DTO 投影将查询结果直接映射到 DTO 类型。不考虑领域类型上的字段映射。直接使用 DTO 类型,您的查询方法可以受益于不受领域模型限制的更动态的投影。
使用多个数据库
当使用多个、可能不同的数据库时,您的应用程序将需要不同的配置方法。提供的 AbstractR2dbcConfiguration 支持类假设一个 ConnectionFactory,从中派生 Dialect。也就是说,您需要自己定义一些 bean 来配置 Spring Data R2DBC 以使用多个数据库。
R2DBC 仓库需要 R2dbcEntityOperations 来实现仓库。不使用 AbstractR2dbcConfiguration 的简单仓库扫描配置如下所示
@Configuration
@EnableR2dbcRepositories(basePackages = "com.acme.mysql", entityOperationsRef = "mysqlR2dbcEntityOperations")
static class MySQLConfiguration {
@Bean
@Qualifier("mysql")
public ConnectionFactory mysqlConnectionFactory() {
return …
}
@Bean
public R2dbcEntityOperations mysqlR2dbcEntityOperations(@Qualifier("mysql") ConnectionFactory connectionFactory) {
DatabaseClient databaseClient = DatabaseClient.create(connectionFactory);
return new R2dbcEntityTemplate(databaseClient, MySqlDialect.INSTANCE);
}
}
请注意,@EnableR2dbcRepositories 允许通过 databaseClientRef 或 entityOperationsRef 进行配置。当连接到多个相同类型的数据库时,使用不同的 DatabaseClient bean 会很有用。当使用方言不同的不同数据库系统时,请改用 @EnableR2dbcRepositories(entityOperationsRef = …) `。