iOS開發(fā) —— ReactiveCocoa詳解

最近一直在研究ReactiveCocoa,現(xiàn)在也來講講ReactiveCocoa中一些基礎(chǔ)類的作用。

ReactiveCocoa作用

在我們iOS開發(fā)過程中,當某些事件響應的時候,需要處理某些業(yè)務(wù)邏輯,這些事件都用不同的方式來處理。比如按鈕的點擊使用actionScrollView滾動使用delegate,屬性值改變使用KVO等系統(tǒng)提供的方式。其實這些事件,都可以通過RAC處理。

RACSiganl

RACSiganl: 信號類,只是表示當數(shù)據(jù)改變時,信號內(nèi)部會發(fā)出數(shù)據(jù),它本身不具備發(fā)送信號的能力,而是交給內(nèi)部一個訂閱者去發(fā)出。

RACSubscriber: 表示訂閱者的意思,用于發(fā)送信號,這是一個協(xié)議,不是一個類,只要遵守這個協(xié)議,并且實現(xiàn)方法才能成為訂閱者。通過create創(chuàng)建的信號,都有一個訂閱者,幫助他發(fā)送數(shù)據(jù)

RACDisposable: 用于取消訂閱或者清理資源,當信號發(fā)送完成或者發(fā)送錯誤的時候,就會自動觸發(fā)它。


 //1.創(chuàng)建信號
    RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        //block調(diào)用時刻:每當有訂閱者訂閱信號,就會調(diào)用block
        
        //2.發(fā)送信號
        [subscriber sendNext:@1];
        
        //如果不再發(fā)送數(shù)據(jù),最好發(fā)送信號完成,內(nèi)部會自動調(diào)用[RACDisposable disposable]取消訂閱
        [subscriber sendCompleted];
        
        return [RACDisposable disposableWithBlock:^{
            //block調(diào)用時刻:當信號發(fā)送完成或者發(fā)送錯誤,就會自動執(zhí)行這個block,取消訂閱
            NSLog(@"信號被銷毀");
        }];
    }];
    
    
    //3.訂閱信號
    [signal subscribeNext:^(id x) {
        //block調(diào)用時刻:每當有信號發(fā)送數(shù)據(jù),就會調(diào)用該方法
        NSLog(@"接收到的數(shù)據(jù):%@",x);
    }];
    

RACSubject與RACReplaySubject

RACSubject:信號提供者,自己可以充當信號,又能發(fā)送信號。subject可以想成是signal的變體,就像NSMutableArray相對于NSArray一樣。它們是非RAC的代碼和RAC代碼之間的橋梁。

RACReplaySubject:重復提供信號類,RACSubject的子類。

RACReplaySubjectRACSubject區(qū)別:

  • RACReplaySubject可以先發(fā)送信號,再訂閱信號,RACSubject就不可以。

  • RACReplaySubject可以設(shè)置capacity數(shù)量來限制緩存的value的數(shù)量,即只緩充最新的幾個值。

  • 如果一個信號每被訂閱一次,就需要把之前的值重復發(fā)送一遍,就需要使用RACReplaySubject


   //1.創(chuàng)建信號
    RACSubject *subject = [RACSubject subject];
    
    //2.訂閱信號
    [subject subscribeNext:^(id x) {
        //block調(diào)用時刻:當信號發(fā)出新值,就會調(diào)用
        NSLog(@"第一個訂閱者%@",x);
    }];
    [subject subscribeNext:^(id x) {
        //block調(diào)用時刻:當信號發(fā)出新值,就會調(diào)用
        NSLog(@"第二個訂閱者%@",x);
    }];
    
    //3.發(fā)送信號
    [subject sendNext:@"1"];
    
  
    
      //1.創(chuàng)建信號
    RACReplaySubject *replaySubject = [RACReplaySubject subject];
   // RACReplaySubject *replaySubject = [RACReplaySubject replaySubjectWithCapacity:0];
    
    //2.發(fā)送信號
    [replaySubject sendNext:@1];
    [replaySubject sendNext:@2];
    
    //3.訂閱信號
    [replaySubject subscribeNext:^(id x) {
        NSLog(@"第一個訂閱者%@",x);
    }];
    [replaySubject subscribeNext:^(id x) {
        NSLog(@"第二個訂閱者%@",x);
    }];

RACSubject替代代理

情景:跳轉(zhuǎn)到另一個ViewController,TwoViewController發(fā)送通知,ViewController收到回調(diào)的通知


//ViewController里

- (IBAction)click:(UIButton *)sender {

    TwoViewController *twoVC = [[TwoViewController alloc] init];
    
    //設(shè)置代理信號
    twoVC.delegateSubject = [RACSubject subject];
    
    //訂閱代理信號
    [twoVC.delegateSubject subscribeNext:^(id x) {
        NSLog(@"點擊了通知按鈕,%@",x);
    }];
    
    //跳轉(zhuǎn)
    [self presentViewController:twoVC animated:YES completion:nil];

}

//TwoViewConrroller里

    if (self.delegateSubject) {
        //發(fā)送信號
        [self.delegateSubject sendNext:@"已跳轉(zhuǎn)到TwoVC"];
    }
    

遍歷數(shù)組字典、字典轉(zhuǎn)模型

RACTuple:元組類,類似NSArray,用來包裝值.

RACSequence: RAC中的集合類,用于代替NSArray,NSDictionary,可以使用它來快速遍歷數(shù)組和字典。


    //第一步: 把數(shù)組轉(zhuǎn)換成集合RACSequence             numbers.rac_sequence
    // 第二步: 把集合RACSequence轉(zhuǎn)換RACSignal信號類    numbers.rac_sequence.signal
    // 第三步: 訂閱信號,激活信號,會自動把集合中的所有值,遍歷出來。    
    // 1.遍歷數(shù)組
    NSArray *numbers = @[@1,@2,@3,@4];
    [numbers.rac_sequence.signal subscribeNext:^(id x) {
        NSLog(@"%@",x);
    }];



    // 2.遍歷字典,遍歷出來的鍵值對會包裝成RACTuple(元組對象)
    NSDictionary *dict = @{@"name":@"xiaoming",@"age":@18};    
    // 解包元組,會把元組的值,按順序給參數(shù)里面的變量賦值
    // 相當于以下寫法
    //        NSString *key = x[0];
    //        NSString *value = x[1];
    [dict.rac_sequence.signal subscribeNext:^(id x) {
        RACTupleUnpack(NSString *name,NSString *age) = x;
        NSLog(@"%@ %@",name,age);
    }];
    
  • 字典轉(zhuǎn)模型

    //1.OC寫法
    NSDictionary *dict1 = @{@"name":@"xiaoming",@"age":@18};
    NSDictionary *dict2 = @{@"name":@"xiaohua",@"age":@20};
    NSArray *arrs = @[dict1,dict2];
    
    NSMutableArray *items = [NSMutableArray array];
    
    for (NSDictionary *dict in arrs) {
        FlagItem *item = [FlagItem flagWithDict:dict];
        [items addObject:item];
    }
    
    //2.RAC寫法
    [arrs.rac_sequence.signal subscribeNext:^(id x) {
        //遍歷RAC字典
        FlagItem *item = [FlagItem flagWithDict:x];
        [items addObject:item];
    }];
    
    //3.高級RAC寫法
    // map:映射的意思,目的:把原始值value映射成一個新值
    // array: 把集合轉(zhuǎn)換成數(shù)組
    // 底層實現(xiàn):當信號被訂閱,會遍歷集合中的原始值,映射成新值,并且保存到新的數(shù)組里。
    NSArray *flags = [[arrs.rac_sequence map:^id(id value) {
    
        return [FlagItem flagWithDict:value];
    
    }] array];
    NSLog(@"%@",flags);
    

RACCommand

創(chuàng)建并訂閱響應action的信號。 通常command是由UI觸發(fā)的,像一個按鈕被點擊時。當command被觸發(fā)時,控件會?自動被禁?。

有數(shù)據(jù)改變使用RACSignal 有事件處理需要RACCommand

RACCommand設(shè)計思想:內(nèi)部signalBlock為什么要返回一個信號,這個信號有什么用。

  • 在RAC開發(fā)中,通常會把網(wǎng)絡(luò)請求封裝到RACCommand,直接執(zhí)行某個RACCommand就能發(fā)送請求。

  • RACCommand內(nèi)部請求到數(shù)據(jù)的時候,需要把請求的數(shù)據(jù)傳遞給外界,這時候就需要通過signalBlock返回的信號傳遞了。

使用場景,監(jiān)聽按鈕點擊,網(wǎng)絡(luò)請求


   //1.創(chuàng)建命令
    RACCommand *command = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
        
        NSLog(@"執(zhí)行命令");
        
        // signalBlock必須要返回一個信號,不能傳nil,如果不想要傳遞信號,直接創(chuàng)建空的信號。
        //return [RACSignal empty];
        
        //2.創(chuàng)建信號,用來傳遞數(shù)據(jù)
        return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
            
            [subscriber sendNext:@"請求數(shù)據(jù)"];
            
            //RACCommand中信號如果數(shù)據(jù)傳遞完,必須調(diào)用[subscriber sendCompleted],這時命令才會執(zhí)行完畢,否則永遠處于執(zhí)行中。
            [subscriber sendCompleted];
        
            return nil;
        }];
    }];
    
    //RACCommand需要被強引用,否則接收不到RACCommand中的信號,因此RACCommand中的信號是延遲發(fā)送的。
    _command = command;
    
    //3.訂閱信號
    [command.executionSignals subscribeNext:^(id x) {
        
        [x subscribeNext:^(id x) {
            NSLog(@"%@",x);
        }];
    }];
    
    //RAC高級用法:
    // switchToLatest:用于signal of signals,獲取signal of signals發(fā)出的最新信號,也就是可以直接拿到RACCommand中的信號,不需要訂閱信號
    [command.executionSignals.switchToLatest subscribeNext:^(id x) {
        NSLog(@"%@",x);
    }];
    
    //監(jiān)聽命令是否執(zhí)行完畢,默認會來一次,可以直接跳過,skip表示跳過第一次命令
    [[command.executing skip:1] subscribeNext:^(id x) {
        
        if ([x boolValue] == YES) {
            NSLog(@"正在執(zhí)行");
        }else{
            NSLog(@"執(zhí)行完成");
        }
    }];
    
    
    //4.執(zhí)行命令
    [self.command execute:nil];
    

RACMulticastConnection

用于當一個信號,被多次訂閱時,為了保證創(chuàng)建信號時,避免多次調(diào)用創(chuàng)建信號中的block,造成副作用,可以使用這個類處理。例如:當有2個RACSignal訂閱信心的時候,就需要發(fā)送兩次RACSiagnal的信號,執(zhí)行兩次block操作。而使用RACMulticastConnection連接,對signal pulish處理就不會多次創(chuàng)建。


 //1.創(chuàng)建信號a
    RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        NSLog(@"發(fā)送請求");
        
        [subscriber sendNext:@1];
        
        return nil;
    }];
    
    //2,創(chuàng)建連接
    RACMulticastConnection *connect = [signal publish];
    
    //3.訂閱信號。即使訂閱了,還沒激活信號
    [connect.signal subscribeNext:^(id x) {
        NSLog(@"訂閱者第一信號");
    }];
    [connect.signal subscribeNext:^(id x) {
        NSLog(@"訂閱者第二信號");
    }];
    
    //4.連接,激活信號
    [connect connect];

ReactiveCocoa的其他用法

  • 代替代理:rac_signalForSelector:用于替代代理。

  • 代替KVO :rac_valuesAndChangesForKeyPath:用于監(jiān)聽某個對象的屬性改變。

  • 監(jiān)聽事件:rac_signalForControlEvents:用于監(jiān)聽某個事件。

  • 代替通知:rac_addObserverForName:用于監(jiān)聽某個通知。

  • 監(jiān)聽文本框文字改變:rac_textSignal:只要文本框發(fā)出改變就會發(fā)出這個信號。

  • 處理當界面有多次請求時,需要都獲取到數(shù)據(jù)時,才能展示界面

  • rac_liftSelector:withSignalsFromArray:Signals:當傳入的Signals(信號數(shù)組),每一個signal都至少sendNext過一次,就會去觸發(fā)第一個selector參數(shù)的方法。使用注意:幾個信號,參數(shù)一的方法就幾個參數(shù),每個參數(shù)對應信號發(fā)出的數(shù)據(jù)。


// 1.代替代理
// 需求:自定義redView,監(jiān)聽紅色view中按鈕點擊
// 之前都是需要通過代理監(jiān)聽,給紅色View添加一個代理屬性,點擊按鈕的時候,通知代理做事情
// rac_signalForSelector:把調(diào)用某個對象的方法的信息轉(zhuǎn)換成信號,就要調(diào)用這個方法,就會發(fā)送信號。
// 這里表示只要redV調(diào)用btnClick:,就會發(fā)出信號,訂閱就好了。
[[redV rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(id x) {
    NSLog(@"點擊紅色按鈕");
}];
// 2.KVO
// 把監(jiān)聽redV的center屬性改變轉(zhuǎn)換成信號,只要值改變就會發(fā)送信號
// observer:可以傳入nil
[[redV rac_valuesAndChangesForKeyPath:@"center" options:NSKeyValueObservingOptionNew observer:nil] subscribeNext:^(id x) {
     
    NSLog(@"%@",x);
     
}];
// 3.監(jiān)聽事件
// 把按鈕點擊事件轉(zhuǎn)換為信號,點擊按鈕,就會發(fā)送信號
[[self.btn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
     
    NSLog(@"按鈕被點擊了");
}];
// 4.代替通知
// 把監(jiān)聽到的通知轉(zhuǎn)換信號
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(id x) {
    NSLog(@"鍵盤彈出");
}];
// 5.監(jiān)聽文本框的文字改變
[_textField.rac_textSignal subscribeNext:^(id x) {
     
    NSLog(@"文字改變了%@",x);
}];
// 6.處理多個請求,都返回結(jié)果的時候,統(tǒng)一做處理.
RACSignal *request1 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
     
    // 發(fā)送請求1
    [subscriber sendNext:@"發(fā)送請求1"];
    return nil;
}];
RACSignal *request2 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    // 發(fā)送請求2
    [subscriber sendNext:@"發(fā)送請求2"];
    return nil;
}];
// 使用注意:幾個信號,參數(shù)一的方法就幾個參數(shù),每個參數(shù)對應信號發(fā)出的數(shù)據(jù)。
[self rac_liftSelector:@selector(updateUIWithR1:r2:) withSignalsFromArray:@[request1,request2]];
}
// 更新UI
- (void)updateUIWithR1:(id)data r2:(id)data1
{
    NSLog(@"更新UI%@  %@",data,data1);
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,908評論 6 541
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,324評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,018評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,675評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,417評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,783評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,779評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,960評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,522評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,267評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,471評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,009評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,698評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,099評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,386評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,204評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,436評論 2 378

推薦閱讀更多精彩內(nèi)容