每周的定期博客。
本周个人认为的亮点之一是在测试中使用RxJava,把异步调用同步化。让代码变得“清晰”了许多。
如果你使用基于异步的编程方式的话,肯定会碰到一个如何测试的问题。最简单的比如说
AsyncResult<String> foo = asyncService.foo();
或者基于回调方式的异步代码
asyncSerivce.foo(asyncResult -> {
// do stuff
});
前一种还好,后一种在多个异步调用时很容易碰到callback hell的问题。
A.foo(a -> {
a.bar(b -> {
b.baz(c -> {
});
});
});
比如说上面这种嵌套比较深的代码。一种解决方法是类似JavaScript的Promise的链式调用。
A.foo() .andThen(a -> a.bar()) .andThen(b -> b.baz())
但是链式调用的一个问题是,在方法调用之间是非简单类型的依赖关系时,你需要一些中间类,而且调用顺序有所变化时,中间类也必须随之改变。比如说上述代码中bar依赖A,baz依赖A与B时
A.foo() .andThen(a -> a.bar().success(b -> (a, b)) .andThen((a, b) -> a.baz(b))
相比之前的写法会显得繁琐。对此,你可以改用JavaScript的async/await,完全把代码改成同步风格,免去些中间类的需要。
async/await是协程的一种实现,但是Java没有协程,该怎么办?方法很简单,利用类似Future的get把代码同步化就可以了。特别是在测试代码中,异步代码转开成同步代码并没有太大问题。相反的,同步风格的测试代码更好理解。
于是在测试代码中可以这么做
- 把回调式代码转为Observable事件源
- 使用Observable的blockGet,blockAwait转换为同步代码
Observable<A> makeSourceA() {
Observable.create(emitter -> {
A.foo(a -> {
if(a.isSuccess()) {
emitter.onNext(a.get());
} else {
emitter.onError(a.getError());
}
emitter.onComplete();
});
});
}
Observable<B> makeSourceB(A a) {
Observable.create(emitter -> {
a.bar(b -> {
if(b.isSuccess()) {
emitter.onNext(b.get());
} else {
emitter.onError(b.getError());
}
emitter.onComplete();
});
});
}
Observable<C> makeSourceC(B b) {
Observable.create(emitter -> {
b.bar(c -> {
if(c.isSuccess()) {
emitter.onNext(c.get());
} else {
emitter.onError(c.getError());
}
emitter.onComplete();
});
});
}
A a = makeSourceA().blockingGet();
B b = makeSourceB(a).blockingGet();
C c = makeSourceC(b).blockingGet();
顺便说一句,如果你不会用RxJava,也没有关系,你可以使用CompletableFuture,也可以达到同样的效果。
希望以上的点子对你有用。