二、基本使用
#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;}];