gpt4 book ai didi

java - RxJava - 当同时使用observeOn和subscribeOn时,ConnectableObservable无法通知其观察者超过128次

转载 作者:行者123 更新时间:2023-12-01 10:35:41 24 4
gpt4 key购买 nike

我有一个使用运行很长时间的 ConnectableObservable 的应用程序。神秘的是,一段时间后,它的观察者停止在其 onNext() 方法中接收通知。

我编写了以下测试来简化示例。它只是一个具有无限循环的 ConnectableObservable,其中一个订阅者同时使用 watchOn 和 subscribeon。调用 128 s.onNext(1) 后,它会停止通知观察者。

@Test
public void testHotObservable() throws InterruptedException{

CountDownLatch latch = new CountDownLatch(1);

ConnectableObservable<Integer> observable = Observable.<Integer>create( (s) -> {
while(true){
try {
Thread.sleep(500);
} catch (Exception e) {
e.printStackTrace();
}
s.onNext(1);
}
})
.observeOn(Schedulers.io())
.subscribeOn(Schedulers.io())
.publish();

Observer<Integer> observer = new Observer<Integer>() {
@Override
public void onNext(Integer i) {
System.out.println("got "+i);
}
@Override
public void onCompleted() {
System.out.println("completed");
}

@Override
public void onError(Throwable e) {
e.printStackTrace();
}
};

observable.subscribe(observer);
observable.connect();

latch.await();
}

这是我调试 RxJava 代码时看到的,我找到了它不调用 Observer 的 onNext() 方法的原因,但我不明白:

1.- s.onNext(1); 被调用:

2.- 执行到达rx.internal.operators.OperatorObserveOn.ObserveOnSubscriber.pollQueue():

void pollQueue() {
int emitted = 0;
final AtomicLong localRequested = this.requested;
final AtomicLong localCounter = this.counter;
do {
localCounter.set(1);
long produced = 0;
long r = localRequested.get();
for (;;) {
...
System.out.println("R: "+r);
if (r > 0) {
Object o = queue.poll();
if (o != null) {
child.onNext(on.getValue(o));
r--;

问题在于 r 的值。第一次执行时,其值始终为 128。每次调用后,它都会减 1 (r--)。这意味着当同时使用observeOn和subscribeOn时,ConnectableObservable只能通知其观察者128次。如果我删除 subscribeOn,r 的值将从每次迭代开始并且有效。

更新:

我找到了解决方案:问题是由.observerOn().subscribeOn()的顺序引起的。如果我将其反转为 .subscribeOn().observeOn() ,它就会起作用(我可以看到 r 的值始终重置为 128)。

无论如何,我希望得到一个解释。

最佳答案

许多异步运算符使用内部固定大小的缓冲区,并依赖于订阅者的频繁请求。就你而言,有些东西没有正确要求,我无法说出它是什么。我建议使用标准组件尝试您的用例,看看可能出了什么问题,即您可以用 PublishSubject + 示例替换您的自定义 Observable:

Subject<Integer, Integer> source = PublishSubject.<Integer>create().toSerialized();

ConnectableObservable<Integer> co = source.sample(
500, TimeUnit.MILLISECONDS, Schedulers.io())
.onBackpressureBuffer().publish();

co.subscribe(yourSubscriber);
co.connect();

source.onNext(1);

关于java - RxJava - 当同时使用observeOn和subscribeOn时,ConnectableObservable无法通知其观察者超过128次,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34749534/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com