首页 > 解决方案 > 使用 Spring Data R2DBC 获取嵌套对象

问题描述

我是 Project Reactor 和 R2DBC 的新手。如何正确使用 Spring Data R2DBC 响应式存储库接收和Flux<Child>合并?Mono<Parent>

家长:

@Table("parent")
public class Parent{

@Id
private Long id;
private String name;

@Transient
private Flux<Child> children;

父存储库:

interface ParentRepository extends ReactiveCrudRepository<Parent, Long> {

@Query()
Mono<Parent> findOneByName(String name);

孩子:

@Table("child")
public class Child{

@Id
private Long id;

子库:

interface ChildRepository extends ReactiveCrudRepository<Child, Long> {

@Query()
Flux<Child> findAllByParentId(Long parentId);

ParentPersistenceAdapter:

public Mono<Parent> findParent(String parentName) {
    return parentRepository.findOneByName(parentName)
      //how to invoke childRepository.findAllByParentId()
      //and set this Flux to Parent's Mono

}

我的解决方案是:

public Mono<Parent> findParent(String parentName) {
    return parentRepository.findOneByName(parentName)
      .map(parent -> {
            Flux<Child> children = childRepository.findAllByParentId(parent.getId())
            children.subscribe();
            return parent.setChildren(children );
        });      
}

标签: javareactive-programmingproject-reactorspring-data-r2dbcr2dbc

解决方案


假设存在withChildren(Flux<Child> children)类型方法,您可以这样做:

parentRepository.findById(parentId)
    .map(p -> p.withChildren(childRepository.findAllByParentId(parentId)));

然而,这有点奇怪——你通常不会有一个Flux像这样的 DAO,因为你需要订阅它并单独管理内容。你通常会有一个List<Child>代替。对于这种情况,您可以将子流收集为列表,zip()将相应的Mono发布者组合在一起,然后将它们组合成您的最终Parent对象。

所以假设一个withChildren(List<Child> children)方法:

Mono.zip(parentRepository.findById(parentId),
    childRepository.findAllByParentId(parentId).collectList(),
    (t1,t2) -> t1.withChildren(t2));

推荐阅读