撰写多个网络调用RxJava - Android

时间:2014-11-02 01:27:21

标签: android system.reactive rx-java

帮助编写多个网络调用并在Rxjava中累积结果。 (我在Android应用程序中使用。)

State 
 -- List<City> cityList;

 City
 - cityId;

RestCall 1
Observable<State> stateRequest = restService.getStates();

RestCall 2
Observable<CityDetail> cityRequest = restService.getCityDetail(cityId);

在UI中,我必须在获取每个城市的所有详细信息后显示城市列表,然后在列表视图中显示。 如何实现parllel网络调用并累积结果。 ?

我希望所有城市细节结果都放在源状态“对象”中的列表中。由于状态对象也有一些需要消除的信息。这可能吗?

stateRequest ??? 
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Subscriber<State>() {
    @Override
    public void onCompleted() {

    }

    @Override
    public void onError(Throwable e) {
    }

    @Override
    public void onNext(State result) {
    // Get city list and display
    }
});

我查看了这个示例,它显示了我们如何能够更多地压缩一个可观察的响应。下面的代码片段显示了3个可观察量。 但在我的情况下,我必须并行或顺序进行20次网络调用(我的意思是在后台但是一个接一个)。我如何实现这一目标。任何帮助或指示?

https://gist.github.com/skehlet/9418379

Observable.zip(f3Observable, f4Observable, f5Observable, new Func3<String, Integer, Integer, Map<String, String>>() {
    @Override
    public Map<String, String> call(String s, Integer integer, Integer integer2) {
        Map<String, String> map = new HashMap<String, String>();
        map.put("f3", s);
        map.put("f4", String.valueOf(integer));
        map.put("f5", String.valueOf(integer2));
        return map;
    }

3 个答案:

答案 0 :(得分:3)

我认为您的代码可以简化为这样的代码,因为您对zip运算符的使用接近使用toList运算符

 stateRequest
 .subscribe(State state ->  {
     Observable.from(state.getCityList())
               .flatMap(City city -> restService.getCityDetail(city.getId())
               .toList()
               .subscribe(List<City> cities -> {

                     state.clear();
                     state.addAll(cities);
               });
     });

由于RxJava没有提供throttle运算符,您可以构建类似这样的内容:

Observable<City> limiter = Observable.zip(Observable.interval(1, SECONDS), aCity, (i, c) -> c);

使用此功能,限制器是一种可观察的,每秒会发出一个城市。

因此,使用您的代码,如果您希望将呼叫限制为getCityDetail,例如:

 Observable<Object> limiter = Observable.interval(1, SECONDS);
 stateRequest
 .subscribe(State state ->  {
     Observable.zip(limiter, Observable.from(state.getCityList()), (i, c) -> c)
               .flatMap(City city -> restService.getCityDetail(city.getId())
               .toList()
               .subscribe(List<City> cities -> {

                     state.clear();
                     state.addAll(cities);
               });
     });

答案 1 :(得分:1)

stateRequest
.flatMap(new Func1<State, Observable<State>>() {
    @Override
    public Observable<State> call(final State state) {

        List<Observable> cityObservablesList = new ArrayList<Observable>();

        for(City city: state.getCityList()) {
            cityObservablesList.add(restService.getCityDetail(city.getId());
        }

        Observable cityObservables = Observable.from(cityObservablesList);
        return Observables.zip(cityObservables, new FuncN<State>() {
            @Override
            public State call(Object... args) {
                List<City> cityList = state.getCityList();
                cityList.clear();
                for(Object object: args) {
                    cityList.add((City)object);
                }

                return state;
            }
        })
    })
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Subscriber<State>() {
    @Override
    public void onCompleted() {

    }

    @Override
    public void onError(Throwable e) {
    }

    @Override
    public void onNext(State result) {
    // Get city list and display
    }
});

我使用zip操作符和Iterable for city list作为第一个参数。 但我面临另一个问题。由于zip并行执行作业,因此并行执行10-15个网络调用,并且服务器拒绝每秒最大查询错误(QPS-403)。 我如何指示zip操作员一个接一个地执行任务?

我确实通过向city observable添加延迟[delay(c * 200,TimeUnit.MILLISECONDS))]来解决此问题。但似乎不是一个合适的解决方案。

有任何建议吗?

答案 2 :(得分:0)

看一下 flatMap(函数..,BiFunction ..)。也许这就是你需要的。

statesRepository.getStates()
   .flatMap(states-> Observable.fromIterable(states))
   .flatMap(
           state-> cityRepository.getStateCities(state),
           (state, cityList) -> {
               state.setCities(cityList);
               return state;
           })
   .subscribe(state-> showStateWithCity(state));