我正在为我的应用开发网络。所以我决定试试Square的Retrofit。我看到它们支持简单的回调
@GET("/user/{id}/photo")
void getUserPhoto(@Path("id") int id, Callback<Photo> cb);
和RxJava的Observable
@GET("/user/{id}/photo")
Observable<Photo> getUserPhoto(@Path("id") int id);
乍一看,两者都非常相似,但当它实现时,它就变得有趣了……
而简单的回调实现看起来类似于:
api.getUserPhoto(photoId, new Callback<Photo>() {
@Override
public void onSuccess() {
}
});
这是非常简单直接的。而使用Observable,它很快就会变得冗长且相当复杂。
public Observable<Photo> getUserPhoto(final int photoId) {
return Observable.create(new Observable.OnSubscribeFunc<Photo>() {
@Override
public Subscription onSubscribe(Observer<? super Photo> observer) {
try {
observer.onNext(api.getUserPhoto(photoId));
observer.onCompleted();
} catch (Exception e) {
observer.onError(e);
}
return Subscriptions.empty();
}
}).subscribeOn(Schedulers.threadPoolForIO());
}
但事实并非如此。你仍然需要做这样的事情:
Observable.from(photoIdArray)
.mapMany(new Func1<String, Observable<Photo>>() {
@Override
public Observable<Photo> call(Integer s) {
return getUserPhoto(s);
}
})
.subscribeOn(Schedulers.threadPoolForIO())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(new Action1<Photo>() {
@Override
public void call(Photo photo) {
//save photo?
}
});
我是不是遗漏了什么?或者在这种情况下使用可观察对象是错误的?
什么时候会/应该更喜欢Observable而不是简单的回调?
更新
使用改造比上面的例子简单得多,就像@Niels在他的回答中或Jake Wharton的例子项目U2020中所展示的那样。但本质上的问题是不变的——什么时候应该使用一种方式或另一种方式?
使用rxjava,你可以用更少的代码做更多的事情。
让我们假设你想在你的应用中实现即时搜索。
通过回调,你担心取消订阅前一个请求并订阅新请求,自己处理方向更改…我认为这是大量的代码和太啰嗦。
用rxjava非常简单。
public class PhotoModel{
BehaviorSubject<Observable<Photo>> subject = BehaviorSubject.create(...);
public void setUserId(String id){
subject.onNext(Api.getUserPhoto(photoId));
}
public Observable<Photo> subscribeToPhoto(){
return Observable.switchOnNext(subject);
}
}
如果你想实现即时搜索,你只需要监听TextChangeListener并调用photommodel . setuserid (EditText.getText());
在Fragment或activity的onCreate方法中,你订阅了返回photomode . subscribetophoto()的Observable,它会返回一个总是发出由最新的Observable(request)发出的项的Observable。
AndroidObservable.bindFragment(this, photoModel.subscribeToPhoto())
.subscribe(new Action1<Photo>(Photo photo){
//Here you always receive the response of the latest query to the server.
});
此外,如果photommodel是一个单例,例如,您不需要担心方向更改,因为不管您何时订阅,BehaviorSubject都会发出最后一个服务器响应。
通过这几行代码,我们实现了即时搜索和处理方向更改。
你认为你可以用更少的代码实现回调吗?我对此表示怀疑。
通过其他答案中的样本和结论,我认为对于简单的一两步任务来说,没有太大的区别。然而,Callback是简单而直接的。RxJava比较复杂,对于简单的任务来说太大了。还有第三种解决办法:算盘常用。让我用这三种解决方案实现上面的用例:Callback, RxJava, CompletableFuture(abacsus -common)和Retrolambda:
从网络获取照片并保存/显示在设备上:
// By Callback
api.getUserPhoto(userId, new Callback<Photo>() {
@Override
public void onResponse(Call<Photo> call, Response<Photo> response) {
save(response.body()); // or update view on UI thread.
}
@Override
public void onFailure(Call<Photo> call, Throwable t) {
// show error message on UI or do something else.
}
});
// By RxJava
api.getUserPhoto2(userId) //
.observeOn(AndroidSchedulers.mainThread())
.subscribe(photo -> {
save(photo); // or update view on UI thread.
}, error -> {
// show error message on UI or do something else.
});
// By Thread pool executor and CompletableFuture.
TPExecutor.execute(() -> api.getUserPhoto(userId))
.thenRunOnUI((photo, error) -> {
if (error != null) {
// show error message on UI or do something else.
} else {
save(photo); // or update view on UI thread.
}
});
并行加载用户详细信息和照片
// By Callback
// ignored because it's little complicated
// By RxJava
Observable.zip(api.getUserDetails2(userId), api.getUserPhoto2(userId), (details, photo) -> Pair.of(details, photo))
.subscribe(p -> {
// Do your task.
});
// By Thread pool executor and CompletableFuture.
TPExecutor.execute(() -> api.getUserDetails(userId))
.runOnUIAfterBoth(TPExecutor.execute(() -> api.getUserPhoto(userId)), p -> {
// Do your task
});