RxJava源码分析(1)
来源:互联网 发布:linux下c 多线程 编辑:程序博客网 时间:2024/05/23 19:58
RxJava源码分析(1)
Rxjava相信大家都不陌生,是现在很流行的一种解决android异步通信的框架,不会对RxJava2的源码逐字逐句的阅读,只寻找关键处,我们平时接触得到的那些代码。
背压实际中接触较少,故只分析了Observable.
分析的源码版本为:2.0.1
我们的目的:
- 知道源头(Observable)是如何将数据发送出去的。
- 知道终点(Observer)是如何接收到数据的。
- 何时将源头和终点关联起来的
- 知道线程调度是怎么实现的
- 知道操作符是怎么实现的
本文先达到目的1 ,2 ,3。
我个人认为主要还是适配器模式的体现,我们接触的就只有Observable和Observer,其实内部有大量的中间对象在适配:将它们两联系起来,加入一些额外功能,例如考虑dispose和hook等。
首先看最基础的应用
Observable<String> observable = Observable.create(new ObservableOnSubscribe<String>() { @Override public void subscribe(ObservableEmitter<String> e) throws Exception { Log.d(TAG,"currentThread="+Thread.currentThread().getName()); e.onNext("11"); e.onComplete(); e.onError(new Throwable("cuowu")); } }); observable.subscribe(new Observer<String>() { @Override public void onSubscribe(Disposable d) { Log.e(TAG,"Disposable="+d.isDisposed()); } @Override public void onNext(String value) { Log.d(TAG,"currentThread="+Thread.currentThread().getName()); Log.e(TAG,"onNext:"+value); } @Override public void onError(Throwable e) { Log.e(TAG,"onError:"+e.getMessage()); } @Override public void onComplete() { Log.e(TAG,"onComplete"); } });
这是我们最普通的使用方式。
再根据使用方法分析源码
从create方法开始分析,首先看里面的参数ObservableOnSubscribe:
public interface ObservableOnSubscribe<T> {void subscribe(ObservableEmitter<T> e) throws Exception;}
可以看到它里面就一个方法subscribe,这个方法正是我们重写的那个方法。再看它里面的参数ObservableEmitter
public interface ObservableEmitter<T> extends Emitter<T> { void setDisposable(Disposable d); void setCancellable(Cancellable c); boolean isDisposed(); ObservableEmitter<T> serialize();}
这几个方法和dispose有关,我们可以暂时不予考虑,接着看看它的父类Emitter
public interface Emitter<T> { void onNext(T value); void onError(Throwable error); void onComplete();}
终于看到我们熟悉的方法了,这就是我们在示例中调用到的几个方法。参数分析完毕,就该来看看具体的流程了
public static <T> Observable<T> create(ObservableOnSubscribe<T> source) { //判null方法我们可以忽略 ObjectHelper.requireNonNull(source, "source is null"); return RxJavaPlugins.onAssembly(new ObservableCreate<T>(source));}
RxJavaPlugins.onAssembly()是一个hook方法,暂时先不去考虑它,也就是说我们需要关注的就是new ObservableCreate(source)这个方法,ObservableCreate是一个继承自Observable的类。
final ObservableOnSubscribe<T> source;public ObservableCreate(ObservableOnSubscribe<T> source) { //在这里就是将ObservableOnSubscribe保存起来 this.source = source;}
至此,整个创建流程结束。再分析订阅的流程:
public final void subscribe(Observer<? super T> observer) { //判null操作 ObjectHelper.requireNonNull(observer, "observer is null"); try { //hook方法 observer = RxJavaPlugins.onSubscribe(this, observer); //判null操作 ObjectHelper.requireNonNull(observer, "Plugin returned null Observer"); //真正执行订阅的方法 subscribeActual(observer); } catch (NullPointerException e) { // NOPMD throw e; } 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); throw npe; }}
从名字可以看出来,真正订阅的地方,但是这里我们需要判断Observable是哪一个了,还记得我们创建返回的是哪个Observable么?就是这个new ObservableCreate(source),下面就该看它的subscribeActual方法了。
protected void subscribeActual(Observer<? super T> observer) { //在这里创建的CreateEmitter实现了Disposable接口,暂时可以看成是一个Disposable,且通过传入的observer将 //observer保存起来 CreateEmitter<T> parent = new CreateEmitter<T>(observer); //首先调用onSubscribe方法,传入Disposable observer.onSubscribe(parent); try { //ObservableOnSubscribe这个源头和下游真正产生联系的方法,这时候源头才开始发送数据 source.subscribe(parent); } catch (Throwable ex) { Exceptions.throwIfFatal(ex); parent.onError(ex); }}
source.subscribe(parent)我们知道我们重写了一下3个方法:
e.onNext("11"); e.onComplete(); e.onError(new Throwable("cuowu"));
这里的e就是parent即CreateEmitter,分别看它的onXXX()方法:
public void onNext(T t) { if (t == null) { onError(new NullPointerException("onNext called with null. Null values are generally not allowed in 2.x operators and sources.")); return; } //在这里判断是否Disposed,当调用了onError和onComplete会设为false,所以执行完这两个方法中任何一个后,源 //头将与下游断开连接,所以onError和onComplete是互斥的。 if (!isDisposed()) { observer.onNext(t); } } @Override public void onError(Throwable t) { if (t == null) { t = new NullPointerException("onError called with null. Null values are generally not allowed in 2.x operators and sources."); } if (!isDisposed()) { try { observer.onError(t); } finally { dispose(); } //通过这个可以知道,当先调用了onComplete方法再调用onError方法就会报错。 } else { RxJavaPlugins.onError(t); } } @Override public void onComplete() { if (!isDisposed()) { try { observer.onComplete(); } finally { dispose(); } } }
到这里一个最简单的流程结束。
总结
- 在subscribeActual()方法中,源头和终点关联起来。
- source.subscribe(parent);这句代码执行时,才开始从发送ObservableOnSubscribe中利用ObservableEmitter发送数据给Observer。即数据是从源头push给终点的。
- CreateEmitter 中,只有Observable和Observer的关系没有被dispose,才会回调Observer的onXXXX()方法
- Observer的onComplete()和onError() 互斥只能执行一次,因为CreateEmitter在回调他们两中任意一个后,都会自动dispose()。根据上一点,验证此结论。
- 先error后complete,complete不显示。 反之会crash
- RxJava源码分析(1)
- RxJava源码初步分析
- RxJava源码分析
- RxJava源码分析
- rxJava 源码 原理分析
- RxJava的源码分析
- Rxjava源码分析
- RxJava源码分析(一)
- RxJava源码分析(二)
- [RxJava学习]subscribeOn源码分析
- [RxJava学习]observeOn源码分析
- RxJava 源码解读分析 Scheduler
- RxJava 源码解读分析 observeOn
- RxJava 源码解读分析 from
- RxJava 源码解读分析 just
- RxJava 源码解读分析 defer
- RxJava 源码解读分析 map
- RxJava 源码解读分析 flatMap
- R语言入门与实践(一)Linux版R语言安装
- 如何检测编译器的大小端
- openstack-nova-创建云主机代码分析
- StaggeredGridLayoutManager 瀑布流返回顶部时跳跃问题 解决办法
- 9-16 acm
- RxJava源码分析(1)
- 程序片段----弹出文件夹
- leetcode 141. Linked List Cycle 链表循环的判定 + 双指针
- 使用Halcon实现图像处理
- openwrt menuconfig恢复默认配置
- java三种方式实现键盘录入
- SpringMVC通过Controller实现页面跳转
- Spring Boots &mybatis连接数据库配置步骤
- docker中使用gitlab笔记