如何在RxJava中使用TestScheduler

Cal*_*lin 16 rx-java

我该如何使用RxJava TestScheduler?我来自.NET背景,但TestSchedulerRxJava似乎与.NET rx中的测试调度程序的工作方式不同.

这是我想测试的示例代码

Observable<Long> tick = Observable.interval(1, TimeUnit.SECONDS);
contactsRepository.find(index)
  .buffer(MAX_CONTACTS_FETCH)
  .zipWith(tick, new Func2<List<ContactDto>, Long, List<ContactDto>>() {
    @Override
    public List<ContactDto> call(List<ContactDto> contactList, Long aLong) {
      return contactList;
    }
  }).subscribe()
Run Code Online (Sandbox Code Playgroud)

我试过了:

subscribeOn(testScheduler)
testScheduler.advanceTimeBy(2, TimeUnit.SECONDS);
testScheduler.triggerActions();
Run Code Online (Sandbox Code Playgroud)

没有运气.

dwu*_*sen 23

我举了一个如何使用a的例子TestScheduler.我认为它与.NET实现非常相似

@Test
public void should_test_the_test_schedulers() {
    TestScheduler scheduler = new TestScheduler();
    final List<Long> result = new ArrayList<>();
    Observable.interval(1, TimeUnit.SECONDS, scheduler).take(5).subscribe(result::add);
    assertTrue(result.isEmpty());
    scheduler.advanceTimeBy(2, TimeUnit.SECONDS);
    assertEquals(2, result.size());
    scheduler.advanceTimeBy(10, TimeUnit.SECONDS);
    assertEquals(5, result.size());
}
Run Code Online (Sandbox Code Playgroud)

https://github.com/bric3/demo-rxjava-humantalk/blob/master/src/test/java/demo/humantalk/rxjava/SchedulersTest.java

编辑 根据您的代码:您应该将调度程序传递给Observable.interval操作,因为这是您要控制的:

    TestScheduler scheduler = new TestScheduler();

    Observable<Long> tick = Observable.interval(1, TimeUnit.SECONDS, scheduler);
    Subscription toBeTested = Observable.from(Arrays.asList(1, 2, 3, 4, 5))
            .buffer(3)
            .zipWith(tick, (i, t) -> i)
            .subscribe(System.out::println);

    scheduler.advanceTimeBy(2, TimeUnit.SECONDS);
Run Code Online (Sandbox Code Playgroud)

  • 是的,这很好,但如何测试一个'真实/一段代码,让我们留下一个方法,一个Observable订阅者执行一些回调onComplete和一些其他回调onError.我在`subscribeOn`中注入了测试调度程序,但没有运气. (3认同)

And*_*sch 12

你有一些课:

public class SomeClass {
  public void someMethod() {
    Observable<Long> tick = Observable.interval(1, TimeUnit.SECONDS);
    contactsRepository.find(index)
      .buffer(MAX_CONTACTS_FETCH)
      .zipWith(tick, new Func2<List<ContactDto>, Long, List<ContactDto>>() {
        @Override
        public List<ContactDto> call(List<ContactDto> contactList, Long aLong) {
          return contactList;
        }
      }).subscribe()
  }
}
Run Code Online (Sandbox Code Playgroud)

[Observable.interval][1]在文档中查找,您将看到它在计算调度程序上运行,因此我们在测试中覆盖它.

public class SomeClassTest {
  private TestScheduler testScheduler;

  @Before
  public void before() {
    testScheduler = new TestScheduler();
    // set calls to Schedulers.computation() to use our test scheduler
    RxJavaPlugins.setComputationSchedulerHandler(ignore -> testScheduler);
  }

  @After
  public void after() {
    // reset it
    RxJavaPlugins.setComputationSchedulerHandler(null);
  }

  @Test
  public void test() {
    SomeClass someInstance = new SomeClass();
    someInstance.someMethod();

    // advance time manually
    testScheduler.advanceBy(1, TimeUnit.SECONDS);
  }
Run Code Online (Sandbox Code Playgroud)

该解决方案是对所接受答案的改进,因为维护了生产代码的质量,完整性和简单性.

  • 谢谢!我正在使用“throttleFirst”,但我的测试失败了,因为它没有像我的 TestScheduler 那样提前时间。 (2认同)