ReactiveCocoa 總結基本使用(二)

二、基本使用

#import// 導入頭文件

2-1、監聽文本框使用

- (void)learnRACWithTextFiled{//? ? // 直接監聽 textFiled的改變//? ? [[self.testTextField rac_signalForControlEvents:UIControlEventEditingChanged] subscribeNext:^(id x){////? ? ? ? NSLog(@"%@", x);//////? ? }];// 或者[self.testTextField.rac_textSignal subscribeNext:^(NSString* textString) {NSLog(@"%@", textString);? ? }];}// 打印出其textFiled中的文本信息來

2-3、 監聽Button事件

- (void)learnRACWithButton{? ? [[self.testButton rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(idx) {NSLog(@"按鈕被點擊了");? ? }];}

2-3、手勢

- (void)learnRACWithGesture{? ? UITapGestureRecognizer * tap =[[UITapGestureRecognizer alloc]init];[self.view addGestureRecognizer:tap];[[tap rac_gestureSignal]subscribeNext:^(UITapGestureRecognizer * tap) {? ? ? ? // 點擊可以[[[UIApplication sharedApplication]keyWindow]endEditing:YES];? ? }];}

2-4、通知

-(void)learnRACWithNSNotificationCenter{// 通知可以不移除[[[NSNotificationCenter defaultCenter]rac_addObserverForName:UIKeyboardWillShowNotificationobject:nil]subscribeNext:^(NSNotification * notification) {NSLog(@"show");? ? }];}

2-5、定時器

-(void)learnRACWithNSTimer{NSLog(@"begin");//? ? 1. 延遲某個時間后再做某件事[[RACScheduler mainThreadScheduler]afterDelay:2.0fschedule:^{NSLog(@"2秒之后發生的事情");? ? }];//? ? 2. 每個一定長度時間做一件事[[RACSignal interval:4 onScheduler:[RACScheduler mainThreadScheduler]]subscribeNext:^(NSDate * date) {NSLog(@"每隔幾秒發生的事情");? ? }];/*

2015-12-21 13:22:23.209 ReactiveCocoaLearn[78775:4675706] begin

2015-12-21 13:22:25.409 ReactiveCocoaLearn[78775:4675706] 2秒之后發生的事情

2015-12-21 13:22:27.213 ReactiveCocoaLearn[78775:4675706] 每隔幾秒發生的事情

2015-12-21 13:22:31.211 ReactiveCocoaLearn[78775:4675706] 每隔幾秒發生的事情

*/}

2-6、代理

但是有局限,只能取代沒有返回值的代理方法

- (void)learnRACWithProtocol{UIAlertView* alertView = [[UIAlertViewalloc]initWithTitle:@"RAC中Protocol"message:@"UIAlertView"delegate:selfcancelButtonTitle:@"Cancel"otherButtonTitles:@"OK",nil];? ? [alertView show];? ? [[selfrac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)]subscribeNext:^(RACTuple*tuple){//可以多嘗試下RACTuple里的屬性NSLog(@"tuple.second == %@",tuple.second);if([tuple.second isEqualToNumber:@0])? ? ? ? {NSLog(@"cancel");? ? ? ? }if([tuple.second isEqualToNumber:@1])? ? ? ? {NSLog(@"ok");? ? ? ? }? ? }];//? ? 更簡單的方式://? ? ? ? [[alertView rac_buttonClickedSignal]subscribeNext:^(id x) {//? ? ? ? ? ? //可以多嘗試下RACTuple里的屬性//? ? ? ? ? ? NSLog(@"%@",x);//? ? ? ? ? ? if([x isEqualToNumber:@0])//? ? ? ? ? ? {//? ? ? ? ? ? ? ? NSLog(@"Cancel");//? ? ? ? ? ? }//? ? ? ? ? ? if([x isEqualToNumber:@1])//? ? ? ? ? ? {//? ? ? ? ? ? ? ? NSLog(@"Ok");//? ? ? ? ? ? }////? ? ? ? }];}

2-7、KVO

[RACObserve(self.testScrollerView, contentOffset) subscribeNext:^(id x) {? ? NSLog(@"Offset=%@",x);}];/*2015-12-2115:06:23.689ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0,0}2015-12-2115:06:23.689ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0,0}2015-12-2115:06:23.711ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0, -1}2015-12-2115:06:23.790ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0, -1.5}2015-12-2115:06:23.870ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0, -2} */

或是

[[self.greenViewrac_valuesAndChangesForKeyPath:@"center"options:NSKeyValueObservingOptionNew observer:nil] subscribeNext:^(idx) {? ? NSLog(@"center===%@",x);}];/* center=== ("NSPoint: {187.5, 333.5}", { kind =1;new ="NSPoint: {187.5, 333.5}";} ) */

以上是一些RAC的基本用法,熟練這幾個以后,我們很多場景都能運用自如,而且會發現RAC真的很方便。

三、RACSignal使用

其實在RAC中最核心的類RACSiganl,搞定這個類就能用ReactiveCocoa開發了。

RACSiganl:信號類,一般表示將來有數據傳遞,只要有數據改變,信號內部接收到數據,就會馬上發出數據。

創建信號 & 激活信號 & 廢棄信號

// 1.創建信號 + (RACSignal *)createSignal:(RACDisposable * (^)(id subscriber))didSubscribe// 2.訂閱信號,才會激活信號. - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock// 3.發送信號 - (void)sendNext:(id)value// 4.廢棄信號? RACDisposable// 創建信號RACSignal *siganl = [RACSignalcreateSignal:^RACDisposable *(id subscriber) {// block調用時刻:每當有訂閱者訂閱信號,就會調用block。// 發送信號[subscribersendNext:@1];// 如果不在發送數據,最好發送信號完成,內部會自動調用[RACDisposable disposable]取消訂閱信號。[subscriber sendCompleted];return[RACDisposabledisposableWithBlock:^{// 銷毀信號// block調用時刻:當信號發送完成或者發送錯誤,就會自動執行這個block,取消訂閱信號。// 執行完Block后,當前信號就不在被訂閱了。NSLog(@"信號銷毀");? ? ? }];? }];// 訂閱信號,才會激活信號.[siganlsubscribeNext:^(id x) {? ? ? ? NSLog(@"接到數據x=%@",x);? ? }];/*

2015-12-21 15:47:18.335 ReactiveCocoaLearn[82287:4789675] 接到數據x=1

2015-12-21 15:47:18.335 ReactiveCocoaLearn[82287:4789675] 信號銷毀

*/

信號的處理

3-1、map

[[self.testTextField.rac_textSignal map:^id(NSString*textStr){? ? return @(textStr.length);}] subscribeNext:^(idx){? ? NSLog(@"x==%@",x);}];// 映射

3-2、filter

[[[self.testTextField.rac_textSignal map:^id(NSString*textStr){? ? return @(textStr.length);}] filter:^BOOL(NSNumber* value){? ? return value.integerValue >2;}] subscribeNext:^(idx){? ? NSLog(@"x==%@",x);}];

過濾掉一部分

3-3、delay

RACSignal *siganl = [[RACSignal createSignal:^RACDisposable *(id subscriber) {NSLog(@"realySendSignal");? ? [subscriber sendNext:@1];? ? [subscriber sendCompleted];return[RACDisposable disposableWithBlock:^{NSLog(@"discard Signal");? ? }];}] delay:3];NSLog(@"SubscriSiganl");[siganl subscribeNext:^(idx) {NSLog(@"recevieSiganl=%@",x);}];// 延遲3秒才接收數據/*

2015-12-21 16:33:05.326 ReactiveCocoaLearn[83488:4831881] 開始預訂信號

2015-12-21 16:33:05.327 ReactiveCocoaLearn[83488:4831881] 真正發送信號

2015-12-21 16:33:05.328 ReactiveCocoaLearn[83488:4831881] 銷毀信號

2015-12-21 16:33:08.621 ReactiveCocoaLearn[83488:4831881] 接收信號=1

*/

注意打印的時間,發送信號,訂閱信號 的時間,再次了解下整個流程。

3-4、startWith

RACSignal *siganl = [[RACSignalcreateSignal:^RACDisposable *(id subscriber) {? ? [subscribersendNext:@"one"];? ? [subscriber sendCompleted];return[RACDisposabledisposableWithBlock:^{? ? }];}]startWith:@"two"];[siganlsubscribeNext:^(id x) {? ? NSLog(@"接收信號=%@",x);}];// 2015-12-21 16:38:27.160 ReactiveCocoaLearn[83642:4836850] 接收信號=two// 2015-12-21 16:38:27.162 ReactiveCocoaLearn[83642:4836850] 接收信號=one

相當于在發送某個信號之前先發送另一個信號

3-5、timeout

RACSignal *siganl = [[RACSignalcreateSignal:^RACDisposable *(id subscriber) {// 假設某個請求的時間用了幾秒[[RACScheduler? mainThreadScheduler]afterDelay:4schedule:^{? ? ? ? [subscribersendNext:@"one"];? ? ? ? [subscriber sendCompleted];? ? }];return[RACDisposabledisposableWithBlock:^{//? ? ? ? ? ? NSLog(@"銷毀信號");}];// 然后timeout就是當超過這個時間的時候就會出錯}]timeout:10.0onScheduler:[RACScheduler mainThreadScheduler]];[siganlsubscribeNext:^(id x){? ? NSLog(@"x==%@",x);}error:^(NSError * error){// 這個地方就很容易來處理錯誤的時候啦NSLog(@"error==%@",[error description]);}completed:^{? ? NSLog(@"completed");}];

比較適合用于 請求超時的時候

3-6、take & skip & takeLast

RACSignal *siganl = [[RACSignalcreateSignal:^RACDisposable *(id subscriber) {? ? [subscribersendNext:@"one"];? ? [subscribersendNext:@"two"];? ? [subscribersendNext:@"three"];? ? [subscribersendNext:@"four"];? ? [subscriber sendCompleted];return[RACDisposable ?disposableWithBlock:^{? ? }];}]take:2];[siganlsubscribeNext:^(id x){? ? NSLog(@"x==%@",x);}];//take 只接收前幾次//skip 跳過前幾次//takeLast 只接收最后幾次/ *takeUntilBlock:takeWhileBlock:skipWhileBlock:skipUntilBlock:*/

四、進階使用

在我們向服務器進行請求的時候,RAC為我們帶來了諸多方便的事情,值得探索。

此處還是用DeveloperLx的例子,textFiled舉例說明。

4-1、throttle

[[self.testTextField.rac_textSignal throttle:0.5]subscribeNext:^(idx){? ? NSLog(@"%@", x);}];

就是在我們設置那個時間內(0.5秒),不會發送消息,讓其不會一直不斷的發送過來。

4-2 distinctUntilChanged

[[[self.testTextField.rac_textSignal throttle:0.5] distinctUntilChanged]subscribeNext:^(idx){? ? NSLog(@"%@", x);}];

相同的就不發送,直到有所該變再發送

4-3 ignore

[[[[self.testTextField.rac_textSignal throttle:0.5] distinctUntilChanged] ignore:@""] subscribeNext:^(idx){? ? NSLog(@"%@", x);}];

忽略某個值,像上面就是忽略 空值

4-4 switchToLatest

先綜合了下 map

[[[[[[self.testTextField.rac_textSignal throttle:0.5] distinctUntilChanged] ignore:@""] map:^id(idvalue){? ? return [RACSignal createSignal:^RACDisposable*(idsubscriber){? ? ? ? [subscriber sendNext:value];[subscriber sendCompleted];return [RACDisposable disposableWithBlock:^{}];}];}]switchToLatest ]subscribeNext:^(NSString* x){? ? NSLog(@"x==%@", x);}];

只執行最后一次,這個地方有待推敲,暫時還不是很理解

4-5 merge

RACSignal * signalA = [RACSignal createSignal:^RACDisposable *(idsubscriber){dispatch_after(dispatch_time(DISPATCH_TIME_NOW,(int64_t)(2* NSEC_PER_SEC)),dispatch_get_main_queue(),^{? ? ? ? [subscribersendNext:@"Signal_A"];[subscribersendCompleted];});return nil;}];RACSignal * signalB = [RACSignal createSignal:^RACDisposable *(idsubscriber){dispatch_after(dispatch_time(DISPATCH_TIME_NOW,(int64_t)(3* NSEC_PER_SEC)),dispatch_get_main_queue(),^{? ? ? ? [subscribersendNext:@"Signal_B"];[subscribersendCompleted];});return nil;}];NSLog(@"開始預訂");[[RACSignal merge:@[signalA, signalB]]subscribeNext:^(idx) {? ? NSLog(@"x==%@",x);}];/*

2015-12-21 17:54:24.105 ReactiveCocoaLearn[85576:4905054] 開始預訂

2015-12-21 17:54:26.306 ReactiveCocoaLearn[85576:4905054] x==Signal_A

2015-12-21 17:54:27.398 ReactiveCocoaLearn[85576:4905054] x==Signal_B

*/

同時訂閱信號

4-6 concat

NSLog(@"開始預訂");[[RACSignal concat:@[signalA, signalB]]subscribeNext:^(id x) {? ? NSLog(@"x==%@",x);}];/*2015-12-2117:57:03.718ReactiveCocoaLearn[85651:4908056] 開始預訂2015-12-2117:57:05.720ReactiveCocoaLearn[85651:4908056] x==Signal_A2015-12-2117:57:09.012ReactiveCocoaLearn[85651:4908056] x==Signal_B*/

執行完A 后才執行 B ,而且A必須成功,B才會執行,他們是異步請求.

4-7、zipwith

NSLog(@"開始預訂");[[signalA zipWith:signalB]subscribeNext:^(id x) {? ? NSLog(@"x==%@",x);}];? /*2015-12-2118:01:18.770ReactiveCocoaLearn[85742:4913279]開始預訂2015-12-2118:01:22.071ReactiveCocoaLearn[85742:4913279]x== ("Signal_A","Signal_B")? */

注意看上面返回的時間差距

返回一個RACTuple(元祖) ,A、B 至少都發送過一次消息后,才返回。

三者以上的可以用下面這個,combineLatest,同上

[[RACSignal combineLatest:@[signalA,signalB,signalC]] subscribeNext:^(idx){? ? NSLog(@"x==%@",x);}];

五、RAC常見宏

5.1RAC(TARGET, [KEYPATH, [NIL_VALUE]]):用于給某個對象的某個屬性綁定

RAC(self.testButton,backgroundColor)= [RACObserve(self.testButton,selected)map:^UIColor *(NSNumber *selected){? ? return [selectedboolValue]? [UIColor redColor] : [UIColor greenColor];}];[[self.testButtonrac_signalForControlEvents:UIControlEventTouchUpInside]subscribeNext:^(UIButton*btn){btn.selected= !btn.selected;}];

直接改變button 的顏色

5.2RACObserve(self, name):監聽某個對象的某個屬性,返回的是信號

[RACObserve(self.greenView, center) subscribeNext:^(idx) {? ? NSLog(@"%@",x);}];

點擊按鈕,改變其center之后

/*2015-12-2118:18:52.229ReactiveCocoaLearn[86031:4931305] NSPoint: {0,0}2015-12-2118:18:54.024ReactiveCocoaLearn[86031:4931305] 按鈕被點擊了2015-12-2118:18:54.025ReactiveCocoaLearn[86031:4931305] NSPoint: {187.5,333.5}*/

下面這個也是同樣的用這個宏的,這是用最少的代碼寫一個秒表。

RAC(self.testLabel, text) =[[RACSignal interval:1 onScheduler:[RACScheduler mainThreadScheduler]]map:^NSString *(NSDate * date) {returndate.description;}];

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • RAC使用測試Demo下載:github.com/FuWees/WPRACTestDemo 1.ReactiveC...
    FuWees閱讀 6,489評論 3 10
  • 我個人非常推崇ReactiveCocoa,它就像中國的太極,太極生兩儀,兩儀生四象,四象生八卦,八卦生萬物。Rea...
    liuyun333閱讀 364評論 0 0
  • 前言 之前對RAC有了一個基本的認識,了解了它的作用,以及RAC的運行機制,我們知道只要是信號(RACSignal...
    大大盆子閱讀 4,518評論 0 11
  • 1.ReactiveCocoa常見操作方法介紹。 1.1 ReactiveCocoa操作須知 所有的信號(RACS...
    萌芽的冬天閱讀 1,042評論 0 5
  • 1.ReactiveCocoa常見操作方法介紹1.1 ReactiveCocoa操作須知所有的信號(RACSign...
    IIronMan閱讀 2,626評論 2 17