【问题标题】:NetworkOnMainThreadException while RxJava CompositeDisposable gets disposedNetworkOnMainThreadException 而 RxJava CompositeDisposable 被处置
【发布时间】:2018-09-11 21:49:30
【问题描述】:

RxJava 还有一个奇怪的问题,我猜它与CompositeDisposable.clear causes OkHttp to throw java.lang.IllegalStateException: Unbalanced enter/exit有关

是同一个问题吗?

代码如下所示:

Observable<Stuff> observable = Observable.create(new ObservableOnSubscribe<Stuff>() {
                @Override
                public void subscribe(@NonNull ObservableEmitter<Stuff> e) throws Exception {

               //do OkHttp stuff, only place with network calls, then call onNext(stuff). 

                }
            })
            .subscribeOn(Schedulers.io())
            .observeOn(AndroidSchedulers.mainThread());
            DisposableObserver<Stuff> disposableObserver = observable
                .subscribeWith(new DisposableObserver<Stuff>() {......});

            disposables.add(disposableObserver);

异常如下所示:

Caused by android.os.NetworkOnMainThreadException
       at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1303)
       at java.net.SocketInputStream.read(SocketInputStream.java:150)
       at java.net.SocketInputStream.read(SocketInputStream.java:120)
       at okio.Okio$2.read(Okio.java:140)
       at okio.AsyncTimeout$2.read(AsyncTimeout.java:237)
       at okio.RealBufferedSource.read(RealBufferedSource.java:47)
       at okhttp3.internal.http1.Http1Codec$AbstractSource.read(Http1Codec.java:363)
       at okhttp3.internal.http1.Http1Codec$FixedLengthSource.read(Http1Codec.java:407)
       at okhttp3.internal.Util.skipAll(Util.java:175)
       at okhttp3.internal.Util.discard(Util.java:157)
       at okhttp3.internal.http1.Http1Codec$FixedLengthSource.close(Http1Codec.java:424)
       at okio.RealBufferedSource.close(RealBufferedSource.java:469)
       at okhttp3.internal.cache.CacheInterceptor$1.close(CacheInterceptor.java:206)
       at okio.RealBufferedSource.close(RealBufferedSource.java:469)
       at okio.RealBufferedSource$1.close(RealBufferedSource.java:453)
       at java.nio.channels.Channels$ReadableByteChannelImpl.implCloseChannel(Channels.java:255)
       at java.nio.channels.spi.AbstractInterruptibleChannel$1.interrupt(AbstractInterruptibleChannel.java:166)
       at java.lang.Thread.interrupt(Thread.java:957)
       at java.util.concurrent.FutureTask.cancel(FutureTask.java:146)
       at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.cancel(ScheduledThreadPoolExecutor.java:258)
       at io.reactivex.internal.schedulers.ScheduledRunnable.dispose(ScheduledRunnable.java:107)
       at io.reactivex.disposables.CompositeDisposable.dispose(CompositeDisposable.java:217)
       at io.reactivex.disposables.CompositeDisposable.dispose(CompositeDisposable.java:80)
       at io.reactivex.internal.schedulers.IoScheduler$EventLoopWorker.dispose(IoScheduler.java:210)
       at io.reactivex.Scheduler$DisposeTask.dispose(Scheduler.java:464)
       at io.reactivex.internal.disposables.DisposableHelper.dispose(DisposableHelper.java:125)
       at io.reactivex.internal.operators.observable.ObservableSubscribeOn$SubscribeOnObserver.dispose(ObservableSubscribeOn.java:74)
       at io.reactivex.internal.operators.observable.ObservableObserveOn$ObserveOnObserver.dispose(ObservableObserveOn.java:146)
       at io.reactivex.internal.disposables.DisposableHelper.dispose(DisposableHelper.java:125)
       at io.reactivex.observers.DisposableObserver.dispose(DisposableObserver.java:91)
       at io.reactivex.disposables.CompositeDisposable.dispose(CompositeDisposable.java:217)
       at io.reactivex.disposables.CompositeDisposable.clear(CompositeDisposable.java:183)
       at mypackage.MyActivity.onStop(MyActivity.java:320)
       at android.app.Instrumentation.callActivityOnStop(Instrumentation.java:1297)
       at android.app.Activity.performStop(Activity.java:7168)
       at android.app.ActivityThread.performDestroyActivity(ActivityThread.java:4543)
       at android.app.ActivityThread.handleDestroyActivity(ActivityThread.java:4609)
       at android.app.ActivityThread.-wrap7(ActivityThread.java)
       at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1692)
       at android.os.Handler.dispatchMessage(Handler.java:102)
       at android.os.Looper.loop(Looper.java:154)
       at android.app.ActivityThread.main(ActivityThread.java:6682)
       at java.lang.reflect.Method.invoke(Method.java)
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1520)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1410)

编辑:这本质上是subscribe 方法中的OkHttp 代码:

try {
            OkHttpClient.Builder okClientBuilder = new OkHttpClient.Builder();
            OkHttpClient client = okClientBuilder.build();
            Request.Builder builder = new Request.Builder()
                    .get()
                    .url(address);
            Call okCall = client.newCall(builder.build());
            Response res = okCall.execute();
            InputStream stream = res.body().byteStream();
            Parser parsed = Parser.parse(stream);
            stream.close();
            e.onNext(parsed);
     } catch (IOException ex) {
        e.onError(ex);
     }

【问题讨论】:

  • 你能把你的DisposableObserver也填上吗?我对正在发生的事情的理论是,您的观察者正在尝试在可观察对象被处置时做某事。
  • 我不会在 DisposableObserver 上覆盖 dispose(),因为它是最终结果。
  • 您仍然替换了您创建的 DisposableObserver 的内容,因此很难缩小问题的范围。
  • 我的onNextonErroronComplete 都不会根据堆栈跟踪被调用,除非我误读了它。似乎它在CompositeDisposable 上调用dispose,这反过来又做了很多事情,包括调用FutureTask.cancel,并触发java.lang.Thread.interrupt。正因为如此,我认为这可能与我在使用完全相同的代码 stackoverflow.com/questions/49581241/… 时遇到的另一个问题有关
  • 请提供“//do OkHttp stuff”和“subscribeWith”的实现。可能是github.com/square/okhttp/issues/1125

标签: android rx-java rx-java2 okhttp okhttp3


【解决方案1】:

您没有显示实际触发问题的代码,但阅读您的堆栈跟踪表明

  • MyActivity.onStop() 被调用,然后
  • CompositeDisposable.clear() 被调用,然后
  • Thread.interrupt() 被调用,然后
  • 网络代码尝试关闭其操作

由于所有这些都是在主线程上完成的,所以抛出异常。

您需要隔离您的网络代码,该代码在您的问题中只有一个带注释的占位符,以便在 IO 线程而不是主线程中启动和停止网络访问。您可能必须使用不同的 observable creator 方法。

【讨论】:

  • 唯一有网络代码的地方是public void subscribe(@NonNull ObservableEmitter&lt;Stuff&gt; e) throws Exception { //do OkHttp stuff, only place with network calls, then call onNext(stuff). }
  • create() 运算符可能不适合处理取消订阅。
猜你喜欢
  • 2021-03-21
  • 2016-03-24
  • 2018-03-21
  • 2018-12-15
  • 2017-01-05
  • 1970-01-01
  • 2017-10-15
  • 2017-09-15
  • 2021-03-25
相关资源
最近更新 更多