java中的 dispose_RxJava2 中多种取消订阅 dispose 的方法梳理( 源码分析 )

Github 相关代码: Github地址

一直感觉 RxJava2 的取消订阅有点混乱, 这样也能取消, 那样也能取消, 没能系统起来的感觉就像掉进了盘丝洞, 迷乱…

下面说说这几种情况

几种取消的情况

subscribe 时返回了 disposable:

6e1eda840b7ce3d442a469167494e015.png

subscribe 不返回 disposable, 从 observer 的 onSubscribe 中获取:

064f69f4ffde5a8d8a052f4eea110dd7.png

之前从网上看的, 使用继承 DisposableObserver 的 observer, 这个 observer 可以直接 dispose

0114a732c9713103c3912f5521d9fc92.png

源码分析

啰嗦啥啊, 这么简单的东西还需要贴源码?

大哥, 下面有总结….

从第一种情况开始看, 我们进入到 .subscribe((s) -> {}) 中看, 发现它是返回了一个四参数的重载方法

public final Disposable subscribe(Consumer super T>onNext) {returnsubscribe(onNext, Functions.ON_ERROR_MISSING, Functions.EMPTY_ACTION, Functions.emptyConsumer());

}

public final Disposable subscribe(Consumer super T> onNext, Consumer super Throwable>onError,

Action onComplete, Consumer super Disposable>onSubscribe) {

...

LambdaObserver ls = new LambdaObserver(onNext, onError, onComplete, onSubscribe);

subscribe(ls);returnls;

}

可以看到, 这个方法里创建了一个 LambdaObserver, 这个 Observer 实现了Disposable 接口, 所以可以直接作为 Disposable 返回到最上级, 这就是为什么第一种情况中的 subscribe 能返回 disposable 的原因.

public final class LambdaObserver extends AtomicReferenceimplements Observer, Disposable, LambdaConsumerIntrospection {

...

}

而第二种情况的 subscribe 其实就是上面方法里的 subscribe(LambdaObserver)

public final void subscribe(Observer super T>observer) {

ObjectHelper.requireNonNull(observer,"observer is null");try{

observer= RxJavaPlugins.onSubscribe(this, observer);

ObjectHelper.requireNonNull(observer,"Plugin returned null Observer");

subscribeActual(observer);

}catch (NullPointerException e) { //NOPMD

throwe;

}catch(Throwable e) {

Exceptions.throwIfFatal(e);//can't call onError because no way to know if a Disposable has been set or not//can't call onSubscribe because the call might have set a Subscription already

RxJavaPlugins.onError(e);

NullPointerException npe= new NullPointerException("Actually not, but can't throw other exceptions due to RS");

npe.initCause(e);thrownpe;

}

}

这个方法内执行了 subscribeActual(observer) 抽象方法, 交给了 Observale 的子类重写.

第三种情况 中出现的 DisposableObserver 与 LambdaObserver 差不多, 甚至更简单

public abstract class DisposableObserver implements Observer, Disposable {

final AtomicReference s = new AtomicReference();

@Overridepublic final voidonSubscribe(@NonNull Disposable s) {if (EndConsumerHelper.setOnce(this.s, s, getClass())) {

onStart();

}

}protected voidonStart() { }

@Overridepublicfinal boolean isDisposed() {return s.get() ==DisposableHelper.DISPOSED;

}

@Overridepublic final voiddispose() {

DisposableHelper.dispose(s);

}

}

简单总结

稍微一看我们就明白了, 当传入 Observer 接口的四个方法时, subscribe 在内部构建了一个 LambdaObserver , 而这个 LambdaObserver 和第三种情况的 DisposableObserver 都实现了 Disposable 接口, 所以可以作为 Disposable 返回, 就是这么简单.

另外第三种情况里出现的 CompositeDisposable, 简单说就是一个 Disposable 集合( 由 RxJava 内部提供的OpenHashSet 维护, 线程安全 ), CompositeDisposable.dispose() 时会遍历内部的所有 Disposable 执行 dispose 操作.

/**

* Dispose the contents of the OpenHashSet by suppressing non-fatal

* Throwables till the end.

* @param set the OpenHashSet to dispose elements of*/

void dispose(OpenHashSet set) {

...for(Object o : array) {if(o instanceof Disposable) {try{

((Disposable) o).dispose();

}catch(Throwable ex) {

Exceptions.throwIfFatal(ex);if (errors == null) {

errors= new ArrayList();

}

errors.add(ex);

}

}

}

...

}

几种方式的适用情况

如果是零星使用的话, 第一种最方便, observer 的四个方法可以按需使用, 相同逻辑的方法有多种可供选择

039df687f4ccf24d470184ea7f1a70dd.png

如果使用的 Observer 有很多共同逻辑, 则可以写一个 BaseObserver 继承 DisposableObserver 或者 LambdaObserver, 直接使用 xxObserver.dispose()

open class BaseObserver() : DisposableObserver()

如果有很多的 diposable 需要取消的话, 使用 CompositeDisposable 会更简单一些

如何在MVP中自动取消订阅避免内存泄漏, 在我的github中有封装示例 Github地址

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值