我想创建以下内容的observable:
我想使用BehaviorSubject<Boolean>
作为触发器,并将此触发器绑定到活动的onResume
和onPause
事件。 (附加代码示例)
问题
我已经设置了一些东西,但它没有按预期工作。我用它如下:
Observable o = ...;
// Variant 1
o = o.lift(new RxValve(getPauser(), 1000, getPauser().getValue())
// Variant 2
// o = o.compose(RXPauser.applyPauser(getPauser()));
o
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe();
目前的问题是,Variant 1应该可以正常工作,但有时,事件只是没有发出 - 阀门没有发射,直到阀门一切正常工作(可能是一个线程问题......)!解决方案2更简单,似乎有效,但我不确定它是否真的更好,我不这么认为。我实际上不确定,为什么解决方案有时会失败所以我不确定解决方案2是否解决了(目前对我来说不为人知)的问题......
有人可以告诉我可能是什么问题,或者简单的解决方案应该可靠地运作吗?或者向我展示一个可靠的解决方案?
代码
RxValue
https://gist.github.com/akarnokd/1c54e5a4f64f9b1e46bdcf62b4222f08
RXPauser功能
public static <T> Observable.Transformer<T, T> applyPauser(Observable<Boolean> pauser)
{
return observable -> pauser(observable, pauser);
}
private static <T> Observable<T> pauser(Observable<T> source, Observable<Boolean> pauser)
{
// this observable buffers all items that are emitted while emission is paused
Observable<T> sharedSource = source.publish().refCount();
Observable<T> queue = sharedSource
.buffer(pauser.distinctUntilChanged().filter(isResumed -> !isResumed), aBoolean -> pauser.distinctUntilChanged().filter(isResumed -> isResumed))
.flatMap(l -> Observable.from(l))
.doOnNext(t -> L.d(RXPauser.class, "Pauser QUEUED: " + t));
// this observable emits all items that are emitted while emission is not paused
Observable<T> window = sharedSource.window(pauser.distinctUntilChanged().filter(isResumed -> isResumed), aBoolean -> pauser.distinctUntilChanged().filter(isResumed -> !isResumed))
.switchMap(tObservable -> tObservable)
.doOnNext(t -> L.d(RXPauser.class, "Pauser NOT QUEUED: " + t));
// combine both observables
return queue.mergeWith(window)
.doOnNext(t -> L.d(RXPauser.class, "Pauser DELIVERED: " + t));
}
活动
public class BaseActivity extends AppCompatActivity {
private final BehaviorSubject<Boolean> pauser = BehaviorSubject.create(false);
public BaseActivity(Bundle savedInstanceState)
{
super(args);
final Class<?> clazz = this.getClass();
pauser
.doOnUnsubscribe(() -> {
L.d(clazz, "Pauser unsubscribed!");
})
.subscribe(aBoolean -> {
L.d(clazz, "Pauser - " + (aBoolean ? "RESUMED" : "PAUSED"));
});
}
public PublishSubject<Boolean> getPauser()
{
return pauser;
}
@Override
protected void onResume()
{
super.onResume();
pauser.onNext(true);
}
@Override
protected void onPause()
{
pauser.onNext(false);
super.onPause();
}
}
答案 0 :(得分:3)
你实际上可以使用.buffer()
运算符传递它可观察,定义何时停止缓冲,从书中采样:
Observable.interval(100, TimeUnit.MILLISECONDS).take(10)
.buffer(Observable.interval(250, TimeUnit.MILLISECONDS))
.subscribe(System.out::println);
您可以使用PublishSubject
作为Observable
在自定义运算符中提供元素。每次需要开始缓冲时,请按Observable.defer(() -> createBufferingValve())
答案 1 :(得分:2)
我为记录事件做了类似的事情
主题收集一些事件,并在10秒内一次将它们推送到服务器。
主要思想是,例如你有班级Event
。
public class Event {
public String jsonData;
public String getJsonData() {
return jsonData;
}
public Event setJsonData(String jsonData) {
this.jsonData = jsonData;
return this;
}
}
您应该为事件创建队列:
private PublishSubject<Event> eventQueue = PublishSubject.create();
它可以是BehaviorSubject
,但这并不重要
然后你应该创建逻辑,它将把事件推送到服务器:
eventObservable = eventQueue.asObservable()
.buffer(10, TimeUnit.SECONDS) //flush events every 10 seconds
.toList()
.doOnNext(new Action1<List<Event>>() {
@Override
public void call(List<Event> events) {
apiClient.pushEvents(events); //push your event
}
})
.onErrorResumeNext(new Func1<Throwable, Observable<List<Event>>>() {
@Override
public Observable<List<Event>> call(Throwable throwable) {
return null; //make sure, that on error will be never called
}
})
.subscribeOn(Schedulers.io())
.observeOn(Schedulers.io());
然后你应该订阅它并保留订阅,直到你不需要它为止:
eventSubscription = eventObservable.subscribe()
首页帮助