RxJava構(gòu)建流基本原理示例解析
正文
本節(jié),我們從Rxjava使用代碼入手,去結(jié)合自己已有的知識(shí)體系,加查閱部分源碼驗(yàn)證的方式,來(lái)一起探索一下Rxjava實(shí)現(xiàn)的基本原理。
為了本文原理分析環(huán)節(jié),可以被更多的人理解、學(xué)習(xí),所以小編從初學(xué)者的角度,從使用入手,一點(diǎn)點(diǎn)的分析了其中的源碼細(xì)節(jié)、思想,建議大家隨著本文的章節(jié)步驟,一步一步的來(lái)閱讀,才能更快、更好的理解Rxjava的真正的思想精髓,也為我們之后的實(shí)踐課程留一個(gè)好的底子。
1.構(gòu)建流
大家一定郁悶,怎么突然跑出來(lái)構(gòu)建流的詞匯,小編你之前也沒(méi)講到過(guò)呀?大家先別急,我們一步一步來(lái)。 首先從一個(gè)最簡(jiǎn)單的,RxJava的樣例使用代碼,我們?nèi)シ纸?,打上步驟
private void test() { //第一步:just調(diào)用 Observable.just("https://img-blog.csdn.net/20160903083319668") //第二步:map調(diào)用 .map(new Function<String, Bitmap>() { @Override public Bitmap apply(String s) throws Exception { //Bitmap bitmap = downloadImage(s); return null; } }) //第三步:subscribeOn、observeOn調(diào)用 .subscribeOn(Schedulers.newThread()) .observeOn(AndroidSchedulers.mainThread()) //第四步:subscribe調(diào)用 .subscribe(new Observer<Bitmap>() { @Override public void onSubscribe() { Log.d(TAG, "onSubscribe"); } @Override public void onNext(Bitmap s) { Log.d(TAG, "onNext s = " + s); } @Override public void onError(Throwable e) { Log.e(TAG, "onError ", e); } @Override public void onComplete() { Log.d(TAG, "onComplete"); } }); }
從上面的樣例代碼分析、分解,我們明面上看到四個(gè)步驟,暫且列下來(lái):
- 第一步:just調(diào)用
- 第二步:map調(diào)用
- 第三步:subscribeOn、observeOn調(diào)用
- 第四步:subscribe調(diào)用
這里運(yùn)行一下,我們看到日志打印
大家看一下這里幾個(gè)關(guān)鍵的函數(shù)對(duì)象
Observable 被觀察者,subscribe訂閱的意思,Observer觀察者的意思,just是僅僅輸入url,subscribeOn和observeOn是線程切換功能,map是中間的一個(gè)操作符。接下來(lái),我們就這些看到的操作符、關(guān)鍵字,進(jìn)行解讀。
1.1 just的解讀
從整體鏈?zhǔn)秸{(diào)用的代碼形式上,我們大概可以猜測(cè)到,just、subscribeOn、observeOn、map等中間操作符,必定都是基于同一個(gè)對(duì)象的變形的builder模式,也就是說(shuō),他們應(yīng)該都是同一個(gè)類(lèi)內(nèi)部的方法。
我們看一下的Observable的just源碼
@CheckReturnValue @NonNull @SchedulerSupport(SchedulerSupport.NONE) public static <T> Observable<T> just(T item) { ObjectHelper.requireNonNull(item, "item is null"); return RxJavaPlugins.onAssembly(new ObservableJust<T>(item)); }
可以看到這里,僅僅是將傳入的T 也就是我們上面樣例中的urlString,包裝為了ObservableJust對(duì)象。而just的調(diào)用,返回的依然是一個(gè)Observable被觀察者對(duì)象。 我們看一下ObservableJust類(lèi)代碼
public final class ObservableJust<T> extends Observable<T> implements ScalarCallable<T> { private final T value; public ObservableJust(final T value) { this.value = value; } @Override protected void subscribeActual(Observer<? super T> observer) { ScalarDisposable<T> sd = new ScalarDisposable<T>(observer, value); observer.onSubscribe(sd); sd.run(); } @Override public T call() { return value; } }
從上面看到,ObservableJust僅僅是將傳入的T封裝了一層而已,它繼承與Observable抽象類(lèi),而Observable抽象類(lèi)實(shí)現(xiàn)了ObservableSource接口
public abstract class Observable<T> implements ObservableSource<T> {
而ObservableSource接口,就是我們外界調(diào)用的subscribe訂閱方法的源頭
public interface ObservableSource<T> { /** * Subscribes the given Observer to this ObservableSource instance. * @param observer the Observer, not null * @throws NullPointerException if {@code observer} is null */ void subscribe(@NonNull Observer<? super T> observer); }
好了,just講到這里,我們繼續(xù)map,到目前為止,ObservableJust的subscribe方法僅僅是一個(gè)方法而已,并未調(diào)用到,所以我們暫時(shí)不予理睬,后面調(diào)用到,我們?cè)偃シ治觥?/p>
1.2 map的解讀
private void test() { //第一步:just調(diào)用 Observable.just("https://img-blog.csdn.net/20160903083319668") //第二步:map調(diào)用 .map(new Function<String, Bitmap>() { @Override public Bitmap apply(String s) throws Exception { //Bitmap bitmap = downloadImage(s); return null; } }) //第三步:subscribeOn、observeOn調(diào)用 .subscribeOn(Schedulers.newThread()) .observeOn(AndroidSchedulers.mainThread()) //第四步:subscribe調(diào)用 .subscribe(new Observer<Bitmap>() { @Override public void onSubscribe() { Log.d(TAG, "onSubscribe"); } @Override public void onNext(Bitmap s) { Log.d(TAG, "onNext s = " + s); } @Override public void onError(Throwable e) { Log.e(TAG, "onError ", e); } @Override public void onComplete() { Log.d(TAG, "onComplete"); } }); }
從這個(gè)使用代碼來(lái)看,map是just與subscribe之前的一個(gè)操作符,起到承上啟下的作用。
我們看到首先是just返回的對(duì)象,可以直接調(diào)用map方法,我們之前分析、實(shí)現(xiàn)過(guò)just,知道just實(shí)際上返回的就是一個(gè)Observable,所以map方法就是Observable里面的一個(gè)方法。
再往下看,map返回后的對(duì)象,直接調(diào)用了subscribe方法,我們之前設(shè)計(jì)實(shí)現(xiàn)響應(yīng)式功能代碼的時(shí)候知道,subscribe是Observable里面的一個(gè)方法,那么這里也就知道了,map方法肯定也是返回一個(gè)Observable對(duì)象。
從上面分析,我們看到一個(gè)流向圖
大家發(fā)現(xiàn),這個(gè)流。只做了一件事情,每個(gè)操作符,都是對(duì)上一層的Observable進(jìn)行了一層包裝代理而已。這時(shí)我們提出一個(gè)概念,這個(gè)流,我們命名為構(gòu)建流。
同理,subscribeOn、observeOn是否也是這樣的呢?我們通過(guò)源碼驗(yàn)證一下。
1.3 subscribeOn、observeOn
有了上面map的經(jīng)驗(yàn),我們猜測(cè),subscribeOn、observeOn順序調(diào)用的時(shí)候,應(yīng)該也僅僅是對(duì)上一層的Observable進(jìn)行了一層包裝代理而已。我們看一下subscribeOn、observeOn的源碼
@CheckReturnValue @SchedulerSupport(SchedulerSupport.CUSTOM) @NonNull public final Observable<T> subscribeOn(@NonNull Scheduler scheduler) { Objects.requireNonNull(scheduler, "scheduler is null"); //果然這里包裝為了ObservableSubscribeOn對(duì)象 return RxJavaPlugins.onAssembly(new ObservableSubscribeOn<>(this, scheduler)); } public final Observable<T> observeOn(@NonNull Scheduler scheduler, boolean delayError, int bufferSize) { Objects.requireNonNull(scheduler, "scheduler is null"); ObjectHelper.verifyPositive(bufferSize, "bufferSize"); return RxJavaPlugins.onAssembly(new ObservableObserveOn<>(this, scheduler, delayError, bufferSize)); }
我們繼續(xù)看一下ObservableSubscribeOn、ObservableObserveOn類(lèi)的源碼
public final class ObservableSubscribeOn<T> extends AbstractObservableWithUpstream<T, T> { final Scheduler scheduler; public ObservableSubscribeOn(ObservableSource<T> source, Scheduler scheduler) { super(source); this.scheduler = scheduler; } //...省略若干 } public final class ObservableObserveOn<T> extends AbstractObservableWithUpstream<T, T> { final Scheduler scheduler; final boolean delayError; final int bufferSize; public ObservableObserveOn(ObservableSource<T> source, Scheduler scheduler, boolean delayError, int bufferSize) { super(source); this.scheduler = scheduler; this.delayError = delayError; this.bufferSize = bufferSize; } }
而AbstractObservableWithUpstream本身就是繼承與Observable
abstract class AbstractObservableWithUpstream<T, U> extends Observable<U> implements HasUpstreamObservableSource<T> {}
1.4 小結(jié)
從上面可以看到,也就是到目前為止,just、map、subscribeOn、observeOn一系列調(diào)用下來(lái),依然都是執(zhí)行一個(gè)操作而已,即每個(gè)操作符,都是對(duì)上一層的Observable進(jìn)行了一層包裝代理而已。
這就是構(gòu)建流,構(gòu)建流的作用就是,從開(kāi)始操作符到訂閱為止,從左往右執(zhí)行,期間每個(gè)操作符,都是對(duì)上一層的Observable進(jìn)行了一層包裝代理而已。如圖:
以上就是RxJava構(gòu)建流基本原理示例解析的詳細(xì)內(nèi)容,更多關(guān)于RxJava構(gòu)建流基本原理的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android自定義Toast樣式實(shí)現(xiàn)方法詳解
這篇文章主要介紹了Android自定義Toast樣式,Toast是一種很方便的消息提示框,會(huì)在 屏幕中顯示一個(gè)消息提示框,沒(méi)任何按鈕,也不會(huì)獲得焦點(diǎn)一段時(shí)間過(guò)后自動(dòng)消失!非常常用!本文就來(lái)通過(guò)一個(gè)例子把Toast的使用講透2023-01-01android應(yīng)用開(kāi)發(fā)之spinner控件的簡(jiǎn)單使用
Android的控件有很多種,其中就有一個(gè)Spinner的控件,這個(gè)控件其實(shí)就是一個(gè)下拉顯示列表。本文通過(guò)腳本之家平臺(tái)給大家介紹android應(yīng)用開(kāi)發(fā)之spinner控件的簡(jiǎn)單使用,感興趣的朋友可以參考下2015-11-11Android利用滑動(dòng)菜單框架實(shí)現(xiàn)滑動(dòng)菜單效果
這篇文章主要介紹了Android實(shí)現(xiàn)滑動(dòng)菜單特效之滑動(dòng)菜單框架完全解析,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05Flutter Component動(dòng)畫(huà)的顯和隱最佳實(shí)踐
這篇文章主要為大家介紹了Flutter Component動(dòng)畫(huà)的顯和隱最佳實(shí)踐詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03Android 瀏覽器的開(kāi)發(fā)實(shí)例分享
本文主要介紹Android 瀏覽器的開(kāi)發(fā),這里提供詳細(xì)的資料及示例代碼,有興趣的小伙伴可以參考下2016-08-08Android利用Paint自定義View實(shí)現(xiàn)進(jìn)度條控件方法示例
這篇文章主要給大家介紹了關(guān)于Android利用Paint自定義View實(shí)現(xiàn)進(jìn)度條控件的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)各位Android開(kāi)發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2017-11-11Android編程開(kāi)發(fā)實(shí)現(xiàn)帶進(jìn)度條和百分比的多線程下載
這篇文章主要介紹了Android編程開(kāi)發(fā)實(shí)現(xiàn)帶進(jìn)度條和百分比的多線程下載,總結(jié)了前面關(guān)于Java多線程下載的技巧,實(shí)例分析了Android實(shí)現(xiàn)帶百分比和進(jìn)度條的多線程下載技巧,需要的朋友可以參考下2015-12-12Android實(shí)現(xiàn)IOS相機(jī)滑動(dòng)控件
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)IOS相機(jī)滑動(dòng)控件的相關(guān)資料,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-08-08TableLayout(表格布局)基礎(chǔ)知識(shí)點(diǎn)詳解
在本文里我們給大家分享了關(guān)于TableLayout(表格布局)的相關(guān)基礎(chǔ)知識(shí)點(diǎn)內(nèi)容,需要的朋友們學(xué)習(xí)下。2019-02-02Android開(kāi)發(fā)之圖片壓縮實(shí)現(xiàn)方法分析
這篇文章主要介紹了Android開(kāi)發(fā)之圖片壓縮實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了Android圖片壓縮的原理、實(shí)現(xiàn)方法及相關(guān)操作注意事項(xiàng),需要的朋友可以參考下2019-03-03