RxJava运算符,在第一个observable发出时立即组合两个observable并发出

时间:2018-01-24 09:50:11

标签: android rx-java

问题: 我有一个功能,用户可以输入一个查询字符串,我创建了2个observable,一个用于查询我的本地数据库,另一个用于从API获取结果。这两个操作必须并行运行。 我需要尽快显示DB的结果,当API结果返回时,我需要检查以删除带有本地结果的重复项。

我的方法: CombineLatest似乎是最接近我需要的东西。但问题是它只在两个观察者都发出结果时才会发出。我认为我正在寻找的是CombineLatest和Concat运算符的混合体。

这是我到目前为止所尝试的内容:

public void performSearchAsync(String query) {
    Observable<List<LocalSearchResult>> localObservable = Observable.just(performLocalSearch(query))
            .subscribeOn(Schedulers.io());
    Observable<Response<List<ApiSearchResult>>> apiObservable = SearchApi.INSTANCE.getSearchResults(query)
            .subscribeOn(Schedulers.io());

    Observable.combineLatest(localObservable, apiObservable, (localSearchResults, listResponse) -> {
        SearchResultWrapper wrapper = new SearchResultWrapper();
        wrapper.localResults = localSearchResults;
        //hold all local result strings in a variable
        List<String> localResultNames = new ArrayList<>();
        for (LocalSearchResult localSearchResult : localSearchResults) {
            localResultNames.add(localSearchResult.name);
        }

        if (!listResponse.isSuccessful()) {
            return wrapper;
        }
        wrapper.apiResults = listResponse.body();

        //Do simple de-dupe
        List<ApiSearchResult> deDupedResults = new ArrayList<>();
        for (ApiSearchResult apiResult : wrapper.apiResults) {
            if (!localResultNames.contains(apiResult.name)) {
                deDupedResults.add(apiResult);
            }
        }
        wrapper.apiResults = deDupedResults;
        return wrapper;
    }).distinctUntilChanged(searchResultWrapper -> searchResultWrapper.localResults)
            .observeOn(AndroidSchedulers.mainThread())
            .subscribe(new Subscriber<SearchResultWrapper>() {
                @Override
                public void onNext(SearchResultWrapper searchResultWrapper) {
                    if (searchResultWrapper == null) return;
                    if (searchResultWrapper.localResults != null && !searchResultWrapper.localResults.isEmpty()) {
                        //add to view
                        applySearchResult(searchResultWrapper.localResults);
                    }
                    if (searchResultWrapper.apiResults == null || searchResultWrapper.apiResults.isEmpty())
                        return;
                    //add to view
                    apiSearchResultsAdapter.updateResults(searchResultWrapper.apiResults);
                }

                @Override
                public void onError(Throwable e) {

                }

                @Override
                public void onCompleted() {

                }
            });
}

static class SearchResultWrapper {
    List<LocalSearchResult> localResults;
    List<ApiSearchResult> apiResults;
}

这里我假设我的本地查询总是比API快,并且在它较慢的可能性中,我希望操作符仅在本地DB observable发出后才发出。我知道重复数据删除逻辑可以改进,它只是一个示例表示,但我希望它发生在我有两个结果的函数中。

即使API observable抛出错误,我也希望发出本地结果并向用户提供,尽管我不确定这种结构是否可行。

希望我的解释足够清楚,有人可以建议一个合适的方法来解决这个问题。我也在考虑编写自己的算子,虽然我听说这很棘手。

1 个答案:

答案 0 :(得分:0)

实际上你不需要太复杂。只需使用mergeconcatdoOnNext即可。

例如(抱歉,我在这里使用kotlin):

    // actually here is a mistake. Observable.just() receive a item rather than a expression.
    // you should use Observable.fromCallable() instead.
    val localObservable = Observable.fromCallable { performLocalSearch(query) }
        .subscribeOn(Schedulers.io())


    val apiObservable = SearchApi.INSTANCE.getFoodSearchResults(query)
        .doOnNext {
            //do your check here
        }
        .subscribeOn(Schedulers.io())
        .map{
            //map it to the same type with local
        }


    localObservable.mergeWith { apiObservable}

或者你可以使用concatWith而不是mergeWith来保证在localObservable Done之后调用你的api调用。