我是RxJava的新手,遇到了以下问题:
我有两个Completable对象用于存储一些数据。我想先触发第一个Completable,然后只有在第一个Completable成功完成后才启动第二个Completable。对第二个Completable的调用应该被阻塞,直到第一个Completable成功完成。此外,如果第一个Completable以错误结束,那么另一个Completable也应该被跳过。
查看文档和其他SO问题,似乎concatWith
或andThen
适合我。但是,在手动测试和单元测试中,我发现第二个Completable与第一个Completable并行触发 :/
第一个Completable
public Completable doA() {
Log.d(TAG, "class call");
return db.countRows()
.doOnSuccess(integer -> {
Log.d(TAG, "found rows: "+integer);
})
.doOnError(Throwable::printStackTrace)
.flatMapCompletable(this::customAction);
}
private Completable customAction(final int count) {
Log.d(TAG, "count: "+count);
if (count > 0) {
Log.d(TAG, "no rows, skip");
return Completable.complete();
}
final User user = ...
return db.save(user); // return Completable
}
第二个可完成的操作
public Completable doB() {
Log.d(TAG, "call to B");
// ...
}
在执行A之后尝试调用B
public Completable someMethod() {
Log.d(TAG, "someMethod");
return doA()
.andThen(doB());
// this also doesn't work
//.concatWith(doB());
}
订阅
someMethod()
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.doOnComplete(() -> {
Log.d(TAG, "complete");
// ...
})
.doOnError(throwable -> {
Log.d("Main", "error "+throwable.getMessage());
// ...
})
.subscribe();
当我运行我的应用并检查日志时,我可以看到:
D/Some method: some method
D/DoA: class call
D/DoB: class call // <- why here?
D/DoA: found rows: 0
D/DoA: count: 0
还有以下单元测试失败:
@Test
public void test() {
when(doa.doA()).thenReturn(Completable.error(new Exception("test")));
observe(); // subscription with TestObserver
verify(dob, never()).doB(); // fails with NeverWantedButInvoked
}
我漏掉了什么?
.andThen(...)
在第一个completable完成后才发生。 - siger