最近一直在研究ReactiveCocoa,現(xiàn)在也來講講ReactiveCocoa中一些基礎(chǔ)類的作用。
ReactiveCocoa作用
在我們iOS開發(fā)過程中,當某些事件響應的時候,需要處理某些業(yè)務(wù)邏輯,這些事件都用不同的方式來處理。比如按鈕的點擊使用action
,ScrollView
滾動使用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的子類。
RACReplaySubject與RACSubject區(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);
}