- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在使用 Retrofit
和 RxJava
,我需要执行以下登录场景:
1) 使用密码登录用户并获取访问 token
2) 获取用户资料,friends(需要分页假设15页),conversations(需要分页假设10页)
第二步的所有请求都需要 token ,因此它们取决于第一步。
我需要做的是执行第一步,完成后执行第二步,并在所有 3 个作业完成时收到通知。
我现在拥有的:
首先,我使用 concatWith
运算符处理分页,并且效果很好。例如:
mApi.getConversationsByPage(accessToken,page,take)
.concatMap(convResponse -> {
if(convResponse.body().getNextPageUrl().equals("NA")) {
return Observable.just(convResponse);
} else {
return Observable.just(convResponse)
.concatWith(getConversationsAndNext(accessToken,convResponse .body().getNextPage(),take));
}
});
我的代码:
mService.loginUser(email,password)
.subscribeOn(mSchedulerProvider.io())
.observeOn(mSchedulerProvider.ui())
.flatMap(tokenResponse -> {
//login finished save token
mService.saveUserToken(tokenResponse.body());
//first step done go to second step
return mService.getUserProfile(mService.getToken();
})
.flatMap(profileResponse -> {
mService.saveUserProfile(profileResponse.body());
return mService.getUserFriendsAndNext(mService.getToken(),FIRST_PAGE,TAKE_PERPAGE);
}).doOnNext(friendResponse -> mService.handleFriendPaging(friendResponse.body().getData()))
.flatMap(friendResponse -> {
mService.saveAllFriends();
return mService.getConversationsAndNext(mService.getToken(),FIRST_PAGE,TAKE_PERPAGE);
})
.subscribe(new Observer<Response<ConvResult>>() {
@Override
public void onSubscribe(Disposable d) {
mDisposables.add(d);
}
@Override
public void onNext(Response<ConvResult> convResponse) {
mService.handleConvPaging(convResponse .body().getData());
}
@Override
public void onError(Throwable e) {
//handle error
}
@Override
public void onComplete() {
//all requests done handle it
mService.saveAllFriends();
}
});
一切正常。我正在保存用户 token 、用户个人资料、用户 friend ,但是当它进入对话时,mService.handleConvPaging
方法调用了 25 次而不是 10 次。
我的测试方法:Mockito.verify(mService,times(10)).handleConvPaging();
错误:
org.mockito.exceptions.verification.TooManyActualInvocations:
mService.handleConvPaging();
Wanted 10 times:
-> at ........
But was 25 times.
所以我想做的是:独立获取用户好友和对话,并在所有请求、分页完成时得到通知。
有没有什么方法可以不使用嵌套回调来实现这一点?提前致谢。
@编辑:
mService.loginUser(email,password)
.flatMap(tokenResponse -> {
if(tokenResponse.isSuccessful()) {
mService.saveUserToken(tokenResponse.body());
return mService.getUserProfile(mService.getUserAccessToken());
} else {
return Observable.error(new Exception("login token error"));
}
})
.flatMap(profileResponse -> {
if(profileResponse.isSuccessful()) {
mService.saveUserProfile(profileResponse.body());
return mService.getUserFriendsAndNext(mService.getUserAccessToken(),1,Constants.TAKE_COUNT);
}else {
return Observable.error(new Exception("login profile error"));
}
})
.doOnNext(friendResponse -> {
if(friendResponse.isSuccessful()) mService.handleFriendPaging(friendResponse.body().getData());
})
.ignoreElements()
.andThen(mService.getConversationsAndNext(mService.getUserAccessToken(),1,Constants.TAKE_COUNT))
.subscribe(onsubscribe,onnext,onerror,oncomplete)
如果我在 tokenResponse
部分返回 Observable.error()
,andThen
方法正在调用,我不希望这种情况发生
而且当 mService.loginUser
方法调用时我也返回这个:
Response<UserTokenResult> tokenResultResponse = Response.error(400,mResponseBody);
最佳答案
正如您在评论中提到的,您的问题是您的第三个 flatMap 应该在第二个 flatMap 完成后调用。所以我想提出这样的解决方案:
mService.loginUser(email,password)
.subscribeOn(mSchedulerProvider.io())
.observeOn(mSchedulerProvider.ui())
.flatMap(tokenResponse -> {
mService.saveUserToken(tokenResponse.body());
return mService.getUserProfile(mService.getToken();
})
.flatMap(profileResponse -> {
mService.saveUserProfile(profileResponse.body());
return mService.getUserFriendsAndNext(mService.getToken(),FIRST_PAGE,TAKE_PERPAGE);
})
// do All your work with your Friends in this operator
.doOnNext(friendResponse -> mService.handleFriendPaging(friendResponse.body().getData()))
//his will return you a Completable
.ignoreElements()
//You get the signal means the upStream works are done
.andThen(mService.getConversationsAndNext(mService.getToken(),FIRST_PAGE,TAKE_PERPAGE))
.subscribe(new Observer<Response<ConvResult>>() {
@Override
public void onSubscribe(Disposable d) {
mDisposables.add(d);
}
@Override
public void onNext(Response<ConvResult> convResponse) {
mService.handleConvPaging(convResponse .body().getData());
}
@Override
public void onError(Throwable e) {
//handle error
}
@Override
public void onComplete() {
//This is somehow not necessary
mService.saveAllFriends();
}
});
因此,您只保留一个流并完成您的工作。
ignoreElements()
提供你只接收终端事件(onComplete() 和 onError())的能力,andThen()
将订阅 upStream Completable 并将继续从您放入 andThen()
的源发出项目。
我测试了类似的东西
我试过类似的东西
Observable.just("A", "B", "C")
.flatMap(x -> Observable.error(new Throwable("Eorror")))
.doOnNext(x -> System.out.println(x))
.ignoreElements()
.andThen(Observable.just("New D"))
.subscribe(x -> System.out.println("onNext" + x),
error -> System.out.println("onError" + error),
() -> System.out.println("END"));
它只是打印
System.out: onErrorjava.lang.Throwable: Eorror
使用 RxJava 版本 2.0.9
关于android - RxJava 和 Retrofit 处理多个平面图的分页,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43683975/
我正在进行 2 个 RX 调用,这些调用相互嵌套且相互依赖。服务器存在问题(由于各种原因目前无法解决),该问题在第二个嵌套调用中返回错误。 在这个问题得到解决之前,我需要确保如果第二次调用返回错误,则
这个问题在这里已经有了答案: How to resolve Duplicate files copied in APK META-INF/rxjava.properties (8 个答案) 关闭 5
我正在尝试将此 RxJava1 代码转换为 RxJava2 public static Observable listFolder(Path dir, String glob) { retur
这个问题在这里已经有了答案: RxJava 1 and RxJava 2 in the same project [duplicate] (1 个回答) How to resolve Duplica
有显示项目的RecyclerViewAdapter(该项目已经映射到数据库中)。 RecyclerViewAdapter 包含对 Presenter 的引用以加载项目。它还包含带有项目 ID 的 Ar
我想弄清楚如何在 Android 中使用 RxJava 将 Realm 对象保存在 Realm 中。到目前为止,结合所有这些的所有示例都是如何从 Realm 读取数据的。我想在 android 中使用
我在日志中收到此错误: Caused by java.lang.ClassCastException: java.net.UnknownHostException cannot be cast to
我有一个 API 服务类,其方法返回 Retrofit 提供的调用。 最近,Rx2Java 引入了 Single,所以我想将 Call 更改为 Single,但我不想更改逻辑。 例如 : 类接口(in
如何使用运算符让我始终获得以前和当前的值?如果可能的话,我想避免在管道外创建状态。 - time -> 1 2 3 4 | | | | Op
我正在努力实现以下目标。我加载了一个对象列表,我想获取稍后放入列表中的值。 首先,我使用 flatmap 将所有值收集到一个数组中(按山顺序),然后当一切完成后,我填充一个适配器。 我无法做的是每
是否可以选择使用 timeout 的变体不发射 Throwable ? 我要 complete事件发出。 最佳答案 您不需要使用 onErrorResumeNext 映射错误。您可以使用以下方法提供备
我们可以在 C# Rx 中异步执行一些代码,如下所示,使用 Observable.Start()。我想知道 RxJava 中的等价物是什么。 void Main() { AddTwoNum
问题:我有一个用户可以输入查询字符串的功能,我制作了 2 个可观察对象,一个用于查询我的本地数据库,另一个用于从 API 获取结果。这两个操作必须并行运行。我需要尽快显示来自数据库的结果,当 API
我正在尝试在 MVVM 中实现 ViewModel,将可观察对象作为“输入流”提供,将观察者作为“输出流”提供以供 View 绑定(bind)。 如果 getUser() 调用成功,下面的代码似乎可以
出于某种原因,我有时想使用 RxOperators 而不是普通的 java 方式来转换数据结构,因为它越来越干净。例如: Observable.from(listOfStrings) .filter(
我是 RxJava 新手,我需要以异步方式使用 Observable 功能。 我还需要使用超时:在我的示例中,我希望每个进程在 1 秒或更短的时间内结束。 这是我现在所做的: public stati
我正在尝试在网络请求期间在UI中显示进度条至少3秒钟。 此答案中描述的相同方法似乎不适用于Single。 RxJava Observable minimum execution time Single
我有一个可观察的(很热),它通过系统进程执行操作,并且我希望也运行一个间隔,直到该进程可观察达到 onComplete。 我看到区间运算符:http://reactivex.io/documentat
好吧,我是 RxJava2 的新手(嗯,我也不了解 RxJava),并且正在尝试使用 RxJava2 和 MVP 结构开发 Android 应用程序。 在该应用程序中,我正在对使用监听器的库进行异步调
如何将单个流拆分为单独的单个流,这样就可以执行以下操作而无需两次计算getUserId()? // getUserId() returns Single getUserId().flatMap { g
我是一名优秀的程序员,十分优秀!