2015-07-09 13:44:04 +08:00
|
|
|
|
RxJava Observables单元测试
|
|
|
|
|
|
---
|
|
|
|
|
|
> * 原文链接 : [Unit Testing RxJava Observables](https://medium.com/ribot-labs/unit-testing-rxjava-6e9540d4a329)
|
|
|
|
|
|
* 原文作者 : [Iván Carballo](https://medium.com/@ivanc)
|
|
|
|
|
|
* [译文出自 : 开发技术前线 www.devtf.cn](http://www.devtf.cn)
|
|
|
|
|
|
* 译者 : [dengshiwei](https://github.com/dengshiwei)
|
2015-07-20 09:09:18 +08:00
|
|
|
|
* 校对者 : [chaossss](https://github.com/chaossss)
|
|
|
|
|
|
* 状态 : 完成
|
2015-07-09 13:44:04 +08:00
|
|
|
|
|
|
|
|
|
|
##RxJava Observables单元测试
|
|
|
|
|
|
|
|
|
|
|
|
RxJava是一个非常棒的类库,但是它不容易上手,在这里,我们列出了不同的方法来对Observables进行单元测试。
|
|
|
|
|
|
|
|
|
|
|
|
在[ribot](http://ribot.co.uk/)时,我们几个月前开始使用RxJava,而现在它已经成为我们开发的安卓应用的核心架构元素。使用RxJava有很多益处,但是学习(使用)的过程是曲折的,很多时候,我们发现我们试图让自己的头脑中环绕着那些解释操作过程如何工作的“美丽”蓝图。
|
|
|
|
|
|
|
|
|
|
|
|
改变我们架构的第一步就是改变那些定义在数据层的方法,让他们返回Observables。下一个问题是:我们如何对此进行单元测试?
|
|
|
|
|
|
|
|
|
|
|
|
###丑陋的方式
|
|
|
|
|
|
|
|
|
|
|
|
我们头脑中首先想到的是以同样的方式简单地描述我们的外部测试结果,然后将结果保存在一个全局变量,可以在以后进行断言。例如,假设我们有一个方法在一个database helper类中,负责加载用户对象。
|
|
|
|
|
|
|
2015-07-17 15:06:38 +08:00
|
|
|
|
```
|
|
|
|
|
|
public Observable<User> loadUser() {
|
2015-07-09 13:44:04 +08:00
|
|
|
|
...
|
2015-07-17 15:06:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
```
|
2015-07-09 13:44:04 +08:00
|
|
|
|
|
|
|
|
|
|
然后,针对这个方法的测试就会如同下面这样:
|
|
|
|
|
|
|
2015-07-17 15:06:38 +08:00
|
|
|
|
```
|
2015-07-09 13:44:04 +08:00
|
|
|
|
User mUser;
|
|
|
|
|
|
@Test
|
|
|
|
|
|
public void shouldLoadUser() throw Exception {
|
|
|
|
|
|
databaseHelper.loadUser()
|
|
|
|
|
|
.subscribe(new Action1<User>() {
|
|
|
|
|
|
@Override
|
|
|
|
|
|
public void call(User user) {
|
|
|
|
|
|
mUser = user;
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
assertNotNull(mUser);
|
|
|
|
|
|
}
|
2015-07-17 15:06:38 +08:00
|
|
|
|
```
|
2015-07-09 13:44:04 +08:00
|
|
|
|
|
|
|
|
|
|
默认情况下,这段代码将会执行,因为Observable会在同一个线程上执行。
|
|
|
|
|
|
因此,在结果设置为全局变量后断言会一直执行。
|
|
|
|
|
|
|
|
|
|
|
|
###更好的方式
|
|
|
|
|
|
我们很快意识到先前的解决方式是不完美的。尽管先前的方法能够工作,(但是)它需要大量的模版代码。所以,我们决定创建一个类,该类提供一个名称为subscribeAssertingThat()的静态方法。这个类允许我们subscribe 一个observable,保存测试结果并且以一种清晰的方式执行一些断言。例如:
|
|
|
|
|
|
|
2015-07-17 15:06:38 +08:00
|
|
|
|
```
|
2015-07-09 13:44:04 +08:00
|
|
|
|
@Test
|
|
|
|
|
|
public void shouldLoadTwoUsers() throw Exception {
|
|
|
|
|
|
subscribeAssertingThat(databaseHelper.loadUser())
|
|
|
|
|
|
.completesSuccessfully()
|
|
|
|
|
|
.hasSize(2)
|
|
|
|
|
|
.emits(user1, user2)
|
|
|
|
|
|
}
|
2015-07-17 15:06:38 +08:00
|
|
|
|
```
|
2015-07-09 13:44:04 +08:00
|
|
|
|
|
|
|
|
|
|
通过100行左右的代码,我们称为RxAssertions的这个类使我们的测试可读性和可写性更好。你可以在[这里](https://gist.github.com/ivacf/874dcb476bfc97f4d555)找到RxAssertions类的代码。
|
|
|
|
|
|
|
|
|
|
|
|
###标准方式
|
|
|
|
|
|
后来,我们发现RxJava 提供了一个称为[TestSubscriber](http://reactivex.io/RxJava/javadoc/rx/observers/TestSubscriber.html#TestSubscriber%28rx.Subscriber%29)的subscriber专用类型。
|
|
|
|
|
|
|
|
|
|
|
|
TestSubscriber由各种各样的[Subscriber](http://reactivex.io/RxJava/javadoc/rx/Subscriber.html)(组成),你可以用于单元测试,执行断言,检查接收到的事件,或封装mocked类型的Subscriber。
|
|
|
|
|
|
|
|
|
|
|
|
与第二个解决方案类似,测试subscriber允许你通过结果执行断言。例如:
|
|
|
|
|
|
|
2015-07-17 15:06:38 +08:00
|
|
|
|
```
|
2015-07-09 13:44:04 +08:00
|
|
|
|
@Test
|
|
|
|
|
|
public void shouldLoadTwoUsers() throw Exception {
|
|
|
|
|
|
TestSubscriber<User> testSubscriber = new TestSubscriber<>();
|
|
|
|
|
|
databaseHelper.loadUser().subscribe(testSubscriber);
|
|
|
|
|
|
testSubscriber.assertNoErrors();
|
|
|
|
|
|
testSubscriber.assertReceivedOnNext(Arrays.asList(user1, user2))
|
|
|
|
|
|
}
|
2015-07-17 15:06:38 +08:00
|
|
|
|
```
|
2015-07-09 13:44:04 +08:00
|
|
|
|
|
|
|
|
|
|
我们还没有频繁的使用TestSubscriber,但你可以看到上面的代码是相当优雅和可读。除了不同的断言,通过 Observable调用getOnNextEvents()方法,它也允许你恢复整个发现的问题列表。
|
|
|
|
|
|
|
|
|
|
|
|
当使用TestSubscriber时,你会发现链接的断言是不可能的,因为断言方法不返回TestSubscriber。这个能力将是一个很好的改善(方向)。
|
|
|
|
|
|
|
|
|
|
|
|
###总结
|
|
|
|
|
|
TestSusbcriber可能是目前最好的选择,因为它是RxJava codebase的一部分,它会不断的改进以及将新功能合并。能够了解别的开发者如何进行Observables的单元测试时很棒的。除了那些我在这里提到的其他方法,你还有什么方法(用于测试)?
|
|
|
|
|
|
|