小编典典

如何在RxJava中进行递归可观察调用?

java

我对RxJava(和一般的Reactive范式)很陌生,所以请多多包涵。

假设我有这个News和这个嵌套的Comment数据结构:

public class News {
  public int id;
  public int[] commentIds; //only top level comments
  public News(int id, int[] commentIds) {
    this.id = id;
    this.commentIds = commentIds;
  }
}

public class Comment {
  public int id;
  public int parentId; //ID of parent News or parent comment
  public int[] childIds;
  public Comment(int id, int parentId, int[] childIds) {
    this.id = id;
    this.parentId = parentId;
    this.childIds = childIds;
  }
}

并假设我有以下API端点:

getComments(int commentId) //return Observable<Comment> for Comment with ID commentId

现在,让我们假设:

getComments(1); //will return Comment(1, 99, [3,4])
getComments(2); //will return Comment(2, 99, [5,6])
getComments(3); //will return Comment(3, 1, [])
getComments(4); //will return Comment(4, 1, [])
getComments(5); //will return Comment(5, 2, [])
getComments(6); //will return Comment(6, 2, [])

**

现在,如果有的话News n = News(99, [1,2]),如何递归所有子项评论?即获得ID为[1,2,3,4,5,6]的评论?

**

我已经搜索并偶然发现了这个:https : //jkschneider.github.io/blog/2014/recursive-
observables-with-
rxjava.html

这是递归函数:

public class FileRecursion {
    static Observable<File> listFiles(File f) {
        if(f.isDirectory())
            return Observable.from(f.listFiles()).flatMap(FileRecursion::listFiles);
        return Observable.just(f);
    }

    public static void main(String[] args) {
          Observable.just(new File("/Users/joschneider/Desktop"))
                  .flatMap(FileRecursion::listFiles)
                  .subscribe(f -> System.out.println(f.getAbsolutePath()));
    }
}

它显示了有关如何进行递归可观察调用的示例,但内部函数(f.listFiles())是阻塞操作(不会返回另一个Observable)。就我而言,内部函数(getComments)是一个非阻塞函数,它返回另一个Observables。我怎么做?

任何帮助都感激不尽。


阅读 244

收藏
2020-11-16

共1个答案

小编典典

这实际上与文章中描述的相同:

Observable<Comment> getInnerComments(Comment comment) {
    if (comment.childIds.length > 0)
        return Observable.merge(
                Observable.just(comment),
                Observable.from(comment.childIds)
                        .flatMap(id -> getComments(id))
                        .flatMap(this::getInnerComments));
    return Observable.just(comment);
}

public static void main(String[] args) {
    getComments(1)
          .flatMap(this::getInnerComments)
          .subscribe(c -> System.out.println(comment.toString()));
}

我先从id =
1的评论开始,然后将其传递给getInnerComments()。该getInnerComments()检查如果评论有孩子。如果是这样,它将遍历每个子ID(Observable#from),并使用您的getComments(int)API
加载每个子ID
。然后,将每个孩子传递给,getInnerComments()以执行相同的过程。如果评论中没有子项,则使用会立即将其返回Observable#just

这是伪代码,尚未经过测试,但是您应该明白这一点。


下面是一个如何获取所有评论,然后将其汇总为一个示例的示例List<Comment>

getNews(99)
        .flatMap(news -> Observable.from(news.commentIds))
        .flatMap(commentId -> getComments(commentId))
        .flatMap(comment -> getInnerComments(comment))
        .toList()
        .subscribe(commentList -> { });
2020-11-16