7

我无法确定Completable在我的测试中是否已订阅 a,例如:

interface IWebApi {
    Observable<Data> download();
}

interface IDataRepository {
    Completable insert(Data data);
}

class SyncService {

    IWebApi webApi;
    IDataRepository repository;

    public SyncService(IWebApi webApi, IDataRepository repository) {
        this.webApi = webApi;
        this.repository = repository;
    }

    public Completable sync() {
        return webApi.download()
            .flatMapCompletable((Data data) -> { repository.insert(data) })
    }
}

然后在我的测试中:

@Test
public void syncTest() {
    Data data = new Data();
    IDataRepository repository = mock (IDataRepository.class);
    IWebApi webApi = mock (IWebApi.class);

    given(webApi.download()).willReturn(Observable.just(data));
    given(repository.insert(data)).willReturn(Completable.complete());

    TestObserver<Void> observer = new TestObserver<Void>();
    SyncService service = new SyncService(webApi, repository);
    service.sync()
            .subscribe(observer);

    observer.assertComplete();
    verify(repository).insert(data);
}

这个测试会通过。但我可以重写同步方法而不使用flatMapCompletable这样的:

    public Completable sync() {
        return webApi.download()
            .doOnNext((Data data) -> { repository.insert(data) })
            .ignoreElements();
    }

然后我的测试将通过,但代码将无法工作,因为即使我认为我已经调用了 insert 方法,但我没有调用subscribe()它。

我应该如何处理这件事?

PS 我是 RxJava 的新手,所以如果我没有使用最佳实践,我很想知道 :)

更新

修复了.ingnoreElements()Maxim Ostrovidov指出的不跟注的错误

4

1 回答 1

7

您可以使用test()运算符来方便:

SyncService service = new SyncService(webApi, repository);
TestObserver observer = service.sync().test();

但是我可以在不使用 flatMapCompletable 的情况下重写同步方法,如下所示:

public Completable sync() {
    return webApi.download()
        .doOnNext((Data data) -> { repository.insert(data) })
}

这不会编译,因为doOnNext仅用于调用项目上的操作,而不是更改流返回类型。在您的情况下,方法是预期的Completable,但实际上会是Observable<Data>.

即使您强制更改最终的流类型:

public Completable sync() {
    return webApi.download()
        .doOnNext((Data data) -> { repository.insert(data) })
        .ignoreElements(); //converts to Completable
}

repository.insert(data)不会被调用,因为doOnNext没有订阅您传递的任何内容并且没有返回任何内容:

//under the hood of lambdas
.doOnNext(new Consumer<Data>() {
    @Override
    public void accept(Data data) throws Exception {

    }
})

您的初始代码最适合您想要实现的目标:

public Completable sync() {
    return webApi.download()
        .flatMapCompletable((Data data) -> { repository.insert(data) })
}

flatMapCompletable订阅通过Completable使用发出的项目Observable

.flatMapCompletable(new Function<Data, CompletableSource>() {
    @Override
    public CompletableSource apply(Data data) throws Exception {
        return repository.insert(data);
    }
})

编辑

要测试repository.insert订阅的事实,您可以使用另一个TestObserverCompletable应用doOnSubscribe

TestObserver repositoryInsertObserver = TestObserver.create();
Completable insertCompletable = Completable.complete()
    .doOnSubscribe(d -> repositoryInsertObserver.onSubscribe(d));

//pass completable to your mock
given(repository.insert(data)).willReturn(insertCompletable);

//and verify that subscription took place
repositoryInsertObserver.assertSubscribed();
于 2017-01-18T01:01:35.423 回答