RAC中常見的高級用法-組合

組合:

concat組合:

按一定順序執(zhí)行皇上與皇太子關系

concat底層實現(xiàn):

1.當拼接信號被訂閱,就會調(diào)用拼接信號的didSubscribe

2.didSubscribe中會先訂閱第一個源信號(signalA)

3.會執(zhí)行第一個源信號(signalA)的didSubscribe

4.第一個源信號(signalA)didSubscribe中發(fā)送值,就會調(diào)用第一個源信號(signalA)訂閱者的nextBlock,通過拼接信號的訂閱者把值發(fā)送出來.

5.第一個源信號(signalA)didSubscribe中發(fā)送完成,就會調(diào)用第一個源信號(signalA)訂閱者的completedBlock,訂閱第二個源信號(signalB)這時候才激活(signalB)。

6.訂閱第二個源信號(signalB),執(zhí)行第二個源信號(signalB)的didSubscribe

7.第二個源信號(signalA)didSubscribe中發(fā)送值,就會通過拼接信號的訂閱者把值發(fā)送出來.

//創(chuàng)建信號A

RACSignal*signalA =[RACSignal createSignal:^RACDisposable*(id subscriber) {

//發(fā)送請求

NSLog(@"上半部分的請求");

//發(fā)送信號

[subscriber sendNext:@"上半部分數(shù)據(jù)"];

//發(fā)送完畢

//加上后就可以上部分發(fā)送完畢后發(fā)送下半部分信號,這個必須要把信號A這個關閉,要不信號B就無法觸發(fā)

[subscriber sendCompleted];

return nil;

}];

//創(chuàng)建信號B

RACSignal*signalB =[RACSignal createSignal:^RACDisposable*(id subscriber) {

//發(fā)送請求

NSLog(@"下半部分的請求");

//發(fā)送信號

[subscriber sendNext:@"下半部分數(shù)據(jù)"];

return nil;

}];

//創(chuàng)建組合信號

// contact:按順序去連接(組合)

//注意:第一個信號必須調(diào)用sendCompleted

RACSignal*contactSignal = [signalA concat: signalB];

//訂閱組合信號

[contactSignal subscribeNext:^(id x) {

//按順序觸發(fā),當A信號觸發(fā)完后,才可使走信號B中的方法,輸出結果可以出結果

NSLog(@"%@",x);

}];

輸出結果:

2016-08-15 10:02:23.060 RAC——oneday[2742:23026]上半部分的請求

2016-08-15 10:02:27.951 RAC——oneday[2742:23026]上半部分數(shù)據(jù)

2016-08-15 10:02:29.806 RAC——oneday[2742:23026]下半部分的請求

2016-08-15 10:02:32.961 RAC——oneday[2742:23026]下半部分數(shù)據(jù)

then:

用于連接兩個信號,當?shù)谝粋€信號完成,才會連接then返回的信號

注意: 使用then之前的信號的值會被忽略掉.

底層實現(xiàn):

1、先過濾掉之前的信號發(fā)出的值。

2.使用concat連接then返回的信號

//創(chuàng)建信號A

RACSignal*signalA = [RACSignalvcreateSignal:^RACDisposable*(id subscriber) {

//發(fā)送請求

NSLog(@"發(fā)送上部分的請求");

//發(fā)送信號

[subscriber sendNext:@"上部分的數(shù)據(jù)"];

//發(fā)送完畢

//加上后就可以發(fā)送上部分的完畢后發(fā)送下部分的信號

[subscriber sendCompleted];

return nil;?

}];

//創(chuàng)建信號B

RACSignal*signalB = [RACSignal createSignal:^RACDisposable*(id subscriber) {

//發(fā)送請求

NSLog(@"發(fā)送下部分的請求");

//發(fā)送信號

[subscriber sendNext:@"下部分的數(shù)據(jù)"];

return nil;

}];

//創(chuàng)建組合信號

//then會忽略點第一個信號的所有值

RACSignal*signalThen = [signalA then:^RACSignal*{

//返回的信號就是需要組合的信號,這里回將signalA信號忽略點

return signalB;

}];

//訂閱信號

[signalThen subscribeNext:^(idx) {

NSLog(@"%@",x);

}];

2016-08-15 10:06:27.643 RAC——oneday[4045:39265]發(fā)送上部分的請求

2016-08-15 10:06:27.645 RAC——oneday[4045:39265]發(fā)送下部分的請求

2016-08-15 10:06:31.261 RAC——oneday[4045:39265]下部分的數(shù)據(jù)

merge

把多個信號合并為一個信號,任何一個信號有新值的時候就會調(diào)用,沒有順序

//創(chuàng)建信號A

RACSubject*siganlA = [RACSubject subject];

//創(chuàng)建信號B

RACSubject*signalB = [RACSubject subject];

//組合信號

RACSubject*signalMerage = [siganlA merge:signalB];

//訂閱信號

[signalMerage subscribeNext:^(id x) {

NSLog(@"%@",x);

}];

//發(fā)送數(shù)據(jù),這里的發(fā)送消息不一定需要哪個信號發(fā)送完畢后才可以,沒有限制

[signalB sendNext:@"下半部數(shù)據(jù)"];

[siganlA sendNext:@"上半部數(shù)據(jù)"];

2016-08-15 10:20:40.660 RAC——oneday[4894:49228]下半部數(shù)據(jù)

2016-08-15 10:20:40.660 RAC——oneday[4894:49228]上半部數(shù)據(jù)

zipWith:

等所有信號都發(fā)送內(nèi)容的時候才會調(diào)用(夫妻關系)

把兩個信號壓縮成一個信號,只有當兩個信號同時發(fā)出信號內(nèi)容時,并且把兩個信號的內(nèi)容合并成一個元組,才會觸發(fā)壓縮流的next事件

底層實現(xiàn):

1.定義壓縮信號,內(nèi)部就會自動訂閱signalA,signalB

2.每當signalA或者signalB發(fā)出信號,就會判斷signalA,signalB有沒有發(fā)出個信號,有就會把最近發(fā)出的信號都包裝成元組發(fā)出。

//創(chuàng)建信號A

RACSubject*signalA = [RACSubject subject];

//創(chuàng)建信號B

RACSubject*signalB = [RACSubject subject];

//壓縮成一個信號

//當一個界面多個請求時,需要等所有的請求都完成才能更新UI

//打印順序跟組合順序,跟發(fā)送的順序無關

//把兩個信號合成一個信號

RACSignal*signalZipwith = [signalA zipWith:signalB];

//訂閱信號

[signalZipwith subscribeNext:^(idx) {

NSLog(@"%@",x);

}];

//發(fā)送信號

[signalA sendNext:@"LUO"];

[signalB sendNext:@"CRAZY"];

2016-08-15 10:32:19.272 RAC——oneday[5618:57722] (

LUO,

CRAZY

)

combineLatest:

將多個信號合并起來,并且拿到各個信號的最新的值,必須每個合并的signal至少都有過一次sendNext,才會觸發(fā)合并的信號。

底層實現(xiàn):

1.當組合信號被訂閱,內(nèi)部會自動訂閱signalA,signalB,必須兩個信號都發(fā)出內(nèi)容,才會被觸發(fā)。

2.并且把兩個信號組合成元組發(fā)出。

//創(chuàng)建信號A

RACSubject*signalA = [RACSubject subject];

//創(chuàng)建信號B

RACSubject*signalB = [RACSubject subject];

//把兩個信號合成一個信號

RACSignal*signalCombineLatest = [signalA combineLatestWith:signalB];

//訂閱組合信號

[signalCombineLatest subscribeNext:^(idx) {

NSLog(@"%@",x);

}];

[signalA sendNext:@"A"];

[signalB sendNext:@"B"];?

2016-08-15 10:43:47.345 RAC——oneday[6427:66814] (

A,

B

)

reduce聚合:用于信號發(fā)出的內(nèi)容是元組,把信號發(fā)出元組的值聚合成一個值

常見的用法(先組合在聚合)。combineLatest:(id)signals reduce:(id (^)())reduceBlock

reduce中的block簡介:

reduceblcok中的參數(shù),有多少信號組合,reduceblcok就有多少參數(shù),每個參數(shù)就是之前信號發(fā)出的內(nèi)容

reduceblcok的返回值:聚合信號之后的內(nèi)容。

底層實現(xiàn):

訂閱聚合信號,每次有內(nèi)容發(fā)出,就會執(zhí)行reduceblcok,把信號內(nèi)容轉(zhuǎn)換成reduceblcok返回的值。

- (void)combineLatestWithReduce

{

/*登錄界面:兩個文本框(賬戶,密碼) + 一個登錄按鈕*/

//組合多個信號

//reduce:聚合

//reduceBlock的參數(shù)與組合的信號一一對應,可以在reduce:后拿到信號的值

RACSignal *combineSignal = [RACSignal combineLatest:@[_accountName.rac_textSignal, _passWord.rac_textSignal] reduce:^id(NSString *account, NSString *pwd){

//block:只要源信號發(fā)送內(nèi)容就會調(diào)用,組合成新的一個值

//聚合的值就是組合信號的內(nèi)容

return @(account.length && pwd.length);

}];

//訂閱信號

//? ? [combineSignal subscribeNext:^(id x) {

//? ? ? ? self.loginBtn.enabled = [x boolValue];

//

//? ? }];

//等同于

RAC(self.loginBtn, enabled) = combineSignal;

}

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容