摘要:信號會由于發(fā)射源的不同發(fā)射到多次而和僅會發(fā)射其中一個,且只發(fā)射一次,標(biāo)志著流的結(jié)束。以此類推直到其中某個流發(fā)出結(jié)束信號,整個被合并后的流結(jié)束,不再發(fā)射數(shù)據(jù)。
前言
forkJoin, zip, combineLatest是rxjs中的合并操作符,用于對多個流進行合并。很多人第一次接觸rxjs時往往分不清它們之間的區(qū)別,其實這很正常,因為當(dāng)你準(zhǔn)備用來合并的流是那種只會發(fā)射一次數(shù)據(jù)就關(guān)閉的流時(比如http請求),就結(jié)果而言這三個操作符沒有任何區(qū)別。
const ob1 = Rx.Observable.of(1).delay(1000); const ob2 = Rx.Observable.of(2).delay(2000); const ob3 = Rx.Observable.of(3).delay(3000); Rx.Observable.forkJoin(ob1, ob2, ob3).subscribe((data) => console.log(data)); Rx.Observable.zip(ob1, ob2, ob3).subscribe((data) => console.log(data)); Rx.Observable.combineLatest(ob1, ob2, ob3).subscribe((data) => console.log(data)); // [1, 2, 3] // [1, 2, 3] // [1, 2, 3] // 都是在3秒的時候打印
rxjs中很多操作符功能相近,只有當(dāng)其操作的流會多次發(fā)射數(shù)據(jù)時才會體現(xiàn)出它們之間的區(qū)別,下面我們來詳細解釋forkJoin, zip, 和combineLatest。
一個基本概念首先我們要知道,一個流(或者說Observable序列)的生命周期中,每次發(fā)射數(shù)據(jù)會發(fā)出next信號(Notification),結(jié)束發(fā)射時會發(fā)出complete信號,發(fā)生錯誤時發(fā)出error信號,三個信號分別對應(yīng)observer的三個方法。next信號會由于發(fā)射源的不同發(fā)射0到多次;而complete和error僅會發(fā)射其中一個,且只發(fā)射一次,標(biāo)志著流的結(jié)束。
subscribe接收一個observer對象用來處理上述三種信號,只傳入一個函數(shù)會被認為是next方法,因此傳入subscribe的next方法會執(zhí)行0到N次,N為序列正常發(fā)射信號的次數(shù)。
用forkJoin合并的流,會在每個被合并的流都發(fā)出結(jié)束信號時發(fā)射一次也是唯一一次數(shù)據(jù)。假設(shè)我們有兩個流:
const ob1 = Rx.Observable.interval(1000).map(d => `ob1:$qoyqs8suu2u`).take(3); const ob2 = Rx.Observable.interval(2000).map(d => `ob2:$qoyqs8suu2u`).take(2); Rx.Observable.forkJoin(ob1, ob2).subscribe((data) => console.log(data)); // ["ob1:2", "ob2:1"]
ob1會在發(fā)射完第三個數(shù)據(jù)時停止發(fā)射,ob2會在發(fā)射完第二個數(shù)據(jù)時停止,而forkJoin合并后的流會等到ob1和ob2都結(jié)束時,發(fā)射一次數(shù)據(jù),也就是觸發(fā)一次subscribe里的回調(diào),接收到的數(shù)據(jù)為ob1和ob2發(fā)射的最后一次數(shù)據(jù)的數(shù)組。
zipzip工作原理如下,當(dāng)每個傳入zip的流都發(fā)射完畢第一次數(shù)據(jù)時,zip將這些數(shù)據(jù)合并為數(shù)組并發(fā)射出去;當(dāng)這些流都發(fā)射完第二次數(shù)據(jù)時,zip再次將它們合并為數(shù)組并發(fā)射。以此類推直到其中某個流發(fā)出結(jié)束信號,整個被合并后的流結(jié)束,不再發(fā)射數(shù)據(jù)。
const ob1 = Rx.Observable.interval(1000).map(d => `ob1:$qoyqs8suu2u`).take(3); const ob2 = Rx.Observable.interval(2000).map(d => `ob2:$qoyqs8suu2u`).take(2); Rx.Observable.zip(ob1, ob2).subscribe({ next: (data) => console.log(data), complete: () => console.log("complete") }); // ["ob1:0", "ob2:0"] ob1等待ob2發(fā)射數(shù)據(jù),之后合并 // ["ob1:1", "ob2:1"] 此時ob2結(jié)束,整個合并的流也結(jié)束 // "complete"
zip和forkJoin的區(qū)別在于,forkJoin僅會合并各個子流最后發(fā)射的一次數(shù)據(jù),觸發(fā)一次回調(diào);zip會等待每個子流都發(fā)射完一次數(shù)據(jù)然后合并發(fā)射,之后繼續(xù)等待,直到其中某個流結(jié)束(因為此時不能使合并的數(shù)據(jù)包含每個子流的數(shù)據(jù))。
combineLatestcombineLatest與zip很相似,combineLatest一開始也會等待每個子流都發(fā)射完一次數(shù)據(jù),但是在合并時,如果子流1在等待其他流發(fā)射數(shù)據(jù)期間又發(fā)射了新數(shù)據(jù),則使用子流最新發(fā)射的數(shù)據(jù)進行合并,之后每當(dāng)有某個流發(fā)射新數(shù)據(jù),不再等待其他流同步發(fā)射數(shù)據(jù),而是使用其他流之前的最近一次數(shù)據(jù)進行合并。
const ob1 = Rx.Observable.interval(1000).map(d => `ob1:$qoyqs8suu2u`).take(3); const ob2 = Rx.Observable.interval(2000).map(d => `ob2:$qoyqs8suu2u`).take(2); Rx.Observable.combineLatest(ob1, ob2).subscribe({ next: (data) => console.log(data), complete: () => console.log("complete") }); // ["ob1:1", "ob2:0"] ob1等待ob2發(fā)射,當(dāng)ob2發(fā)射時ob1已經(jīng)發(fā)射了第二次數(shù)據(jù),使用ob1的第二次數(shù)據(jù) // ["ob1:2", "ob2:0"] ob1繼續(xù)發(fā)射第三次也是最后一次數(shù)據(jù),ob2雖然還未發(fā)射,但是可以使用它上一次的數(shù)據(jù) // ["ob1:2", "ob2:1"] ob2發(fā)射第二次也是最后一次數(shù)據(jù),使ob1上一次的數(shù)據(jù)。 // "complete"
本期內(nèi)容結(jié)束,下一期會繼續(xù)帶來rxjs的其他操作符或者概念詳解。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/92383.html
摘要:原文各種各樣的操作符按照不同的目的,分類幾個大類創(chuàng)建,變化,過濾,組合,廣播,錯誤處理,使用工具等等。 原文:http://reactivex.io/rxjs/manu... 各種各樣的操作符按照不同的目的,分類幾個大類:創(chuàng)建,變化,過濾,組合,廣播(multicasting),錯誤處理,使用工具等等。 以下的列表,按照分類羅列了全部的操作符: 創(chuàng)建操作符 Creation Opera...
摘要:有哪些新變化于年月日正式發(fā)布,為開發(fā)人員帶來了一些令人興奮的增補和改進。不要移除包,直到你將所有的鏈?zhǔn)讲僮餍薷臑楣艿啦僮鞣? RxJS 6有哪些新變化? RxJs 6于2018年4月24日正式發(fā)布,為開發(fā)人員帶來了一些令人興奮的增補和改進。Ben Lesh, rxJS核心開發(fā)成員,強調(diào): RxJS 6在擁有更小API的同時,帶來了更整潔的引入方式 提供一個npm包,該package可...
摘要:輸出流只有在所有的輸入流都完成以后才能完成,任何一條輸入流上的錯誤都將立即推送到輸出流上。如果沒有轉(zhuǎn)入輸入流,輸出流將會立即發(fā)出結(jié)束通知。返回值以數(shù)組形式獲取到的每一個輸入流的值,或者來自映射函數(shù)的值。返回值僅從最新的內(nèi)部流上取值的流。 接著上一節(jié)的操作符繼續(xù),讓我們直奔主題。 組合類操作符 組合類的操作符可以將不同流數(shù)據(jù)按一定的規(guī)則進行合并,從而獲得所需要的完整數(shù)據(jù)。 combine...
摘要:是一種針對異步數(shù)據(jù)流編程工具,或者叫響應(yīng)式擴展編程可不管如何解釋其目標(biāo)就是異步編程,引入為了就是讓異步可控更簡單。最合理的方式在調(diào)用它。當(dāng)組件需要向組件傳遞數(shù)據(jù)時,我們可以在組件中使用。的作用是使指令或組件能自定義事件。 RxJS是一種針對異步數(shù)據(jù)流編程工具,或者叫響應(yīng)式擴展編程;可不管如何解釋RxJS其目標(biāo)就是異步編程,Angular引入RxJS為了就是讓異步可控、更簡單。 而今就是...
摘要:加上以后的操作符大都是直接將輸入流映射到一個輸出流,并且它們都不關(guān)心輸入流上的值。如果輸入流沒發(fā)出任何值,只發(fā)出完成通知,那么發(fā)出一個默認值。與錯誤相關(guān)的一些操作符,如已經(jīng)提到的當(dāng)輸入流上有錯誤時,可以發(fā)出重試,傳入的參數(shù)就是重試的次數(shù)。 這個系列不知不覺已經(jīng)寫到10了,單純從使用上來說的話,大部分的知識點也都講過了,本來不打算寫了,剛好今天有同學(xué)在群里說希望能總結(jié)一下常用的操作符,那...
閱讀 855·2021-11-15 17:58
閱讀 3658·2021-11-12 10:36
閱讀 3794·2021-09-22 16:06
閱讀 969·2021-09-10 10:50
閱讀 1333·2019-08-30 11:19
閱讀 3317·2019-08-29 16:26
閱讀 942·2019-08-29 10:55
閱讀 3349·2019-08-26 13:48