ReactiveCocoa使用个人总结
ReactiveCocoa简介
ReactiveCocoa(简称RAC)是由Github开源的一个应用于iOS和OS开发的新框架。
ReactiveCocoa的作用
在我们iOS开发过程中,当某些事件响应的时候,需要处理某些业务逻辑,这些事件都会用不同的方式来处理。比如:按钮的点击使用action、UIScrollView的滚动使用的delegate、监测属性的值改变使用KVO等系统提供的方式。
RAC就是把这些不同的响应事件的方式统一起来,使这些方式都可以通过RAC来处理。
RAC为事件提供了很多的处理方法,而且利用RAC来处理事件很方便,可以把要处理的事情和监听的代码放一起,方便我们的管理,这样就不需要跳转到对应的方法里去处理,非常符合我们开发中高内聚,低耦合的思想。
RAC在某些特定情况下开发时可以大大简化代码,提高开发效率,并且是安全可靠的。
配置RAC环境
使用cocoapods来获取github上的开源库:
pod 'ReactiveCocoa'
如何使用RAC
- RACSignal
信号类,一般表示将来会有数据传递,只要数据改变,信号内部接收到数据,就会马上发出数据。
- RACSignal只是表示当数据改变时,信号内部会发出数据,他本身不具备发送信号的能力,而是交给内部的一个订阅者
subscriber
发出。
- 默认一个信号都是冷信号,也就是说值改变了也不会触发,只有订阅了这个信号,值改变了才会触发。
- 订阅信号:调用RACSignal的
subscribeNext:
方法就可以订阅信号。
// 创建一个RACSignal信号
RACSignal * signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
// 每当有订阅者订阅信号 就会调用这个block
// 发送信号
[subscriber sendNext:@"Hello"];
// 如果不再发送数据,最好调用此方法发送信号完成
[subscriber sendCompleted];
return [RACDisposable disposableWithBlock:^{
// 当信号发送完成或者发送错误,就会自动执行这个block,取消订阅信号
NSLog(@"信号被销毁");
}];
}];
// 订阅信号,才会激活信号
[signal subscribeNext:^(id x){
// 每当有信号发出数据 就会调用block
NSLog(@"接收到数据%@",x);
}]
- target-action
RAC最基本的使用就是对事件的监听。我们来看以下代码:
RACSignal * textFieldSignal = [textField rac_signalForControlEvents:UIControlEventEditingChanged];
[textFieldSignal subscribeNext:^(id x) {
NSLog(@"变化的text:%@",x);
}];
这段代码实现了监听UITextField
的UIControlEventEditingChanged
事件,当textField
里的内容发生变化时,发送变化的数据调用block种的内容将变化的内容实时打印出来。这两段代码通常会连在一起写:
[[textField rac_signalForControlEvents:UIControlEventEditingChanged] subscribeNext:^(id x) {
NSLog(@"变化的text:%@",x);
}];
同样的,我们以这段代码为模板进行RAC的使用,UIButton
的点击事件我们也可以用RAC进行监测,添加方法,对于UITextField
的文字更改的监听也有更简单的写法:
// UIButton添加RAC监听点击方法
[[button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
NSLog(@"点击button:%@",x);
}];
// 对于UITextField 文字更改的监听
[textField.rac_textSignal subscribeNext:^(id x) {
NSLog(@"变化的text:%@",x);
}]
RAC的一些常见操作方法
-
ReactiveCocoa操作方法之映射(flattenMap,Map)
-
flattenMap
,Map
用于把源信号内容映射成新的内容。
flattenMap的使用:把源信号的内容映射成一个新的信号,信号可以是任意类型
flattenMap
中的block返回的是信号 -
```
// 创建信号中的信号
RACSubject * signalOfSignals = [RACSubject subject];
RACSubject * signals = [RACSubject subject];
[[signalOfSignals flattenMap:^RACStream *(id value) {
// 当signalOfsignals 发出signals 信号才会调用 这里的value返回的是信号
return value;
}] subscribeNext:^(id x) {
// 只有signalOfSignals发出的signals也发出了内容(也就是最后一行代码执行)才会调用
NSLog(@"=========>%@",x);
}];
// 信号中的信号发送一个信号
[signalOfSignals sendNext:signals];
// 信号发送一个内容
[signals sendNext:@"信号发送信号"];
```
> map的使用:把源信号的值映射成一个新的值,`map`中的block返回的是一般对象(这里一般用于把源信号里的json数据映射成相应的model)
```
// 监听文本框内容的改变,映射成一个新的值
[[textField.rac_textSignal map:^id(id value) {
// 这里是把文本框的内容加上 “Hello RAC” 的前缀
return [NSString stringWithFormat:@"Hello RAC :%@",value];
}] subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
```
> 开发中,如果信号发出的不是信号,映射一般使用`map`,如果信号发出的是信号,则使用`flattenMap`。
-
ReactiveCocoa操作方法之组合
-
concat
:按一定顺序拼接信号,当多个信号发出的时候,有顺序的接收信号
RACSignal * signalA = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { [subscriber sendNext:@"First"]; [subscriber sendCompleted]; return nil; }]; RACSignal * signalB = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { [subscriber sendNext:@"Second"]; [subscriber sendCompleted]; return nil; }]; // 把A拼接到B后 A发送完成B才会被激活 按顺序 RACSignal * concatSignal = [signalA concat:signalB]; [concatSignal subscribeNext:^(id x) { NSLog(@"Concat Signal %@",x); }];
-
then
:用于连接两个信号,当第一个信号完成,才会连接then
返回的信号,第一个信号会被忽略掉。
RACSignal * thenSignal = [signalA then:^RACSignal *{ return signalB; }]; [thenSignal subscribeNext:^(id x) { // 此处输出为 Then signal Second 第一个信号会被忽略掉 NSLog(@"Then signal %@",x); }];
-
merge
把多个信号合并为一个信号,任何一个信号有新值就会调用。
RACSignal * mergeSignal = [signalA merge:signalB]; [mergeSignal subscribeNext:^(id x) { NSLog(@"Merge Signal %@",x); }];
-
zipWith
:用于把两个信号压缩成一个信号,只有两个信号都发出信号内容时,把两个信号的内容合并成一个元组,才会触发压缩流的next事件。
RACSignal * zipWithSignal = [signalA zipWith:signalB]; [zipWithSignal subscribeNext:^(id x) { NSLog(@"ZipWith Signal %@",x); }];
-
combineLatest
:将多个信号合并起来,并且拿到各个信号的最新的值,必须每个合并的signal至少都有过一次sendNext,才会触发合并的信号。这个方法试了一下,觉得跟zipWith
是相似的。
RACSignal * combineSignal = [signalA combineLatestWith:signalB]; [combineSignal subscribeNext:^(id x) { NSLog(@"Combine Signal %@",x); }];
-
reduce
:聚合:用于信号发出的内容是元组,把信号发出元组的值聚合成一个值
// 常见的方法(先组合再聚合)。 // 先组合再聚合,把上面两个信号聚合在一起。形成一个信号 (两者都成立才为yes) // 用于登录页面,两个文本框的内容都符合条件,登录按钮才会响应 RACSignal * reduceSignal = [RACSignal combineLatest:@[usernameLengthSignal, passwordLengthSignal] reduce:^id(NSNumber * username , NSNumber * password){ return @(username.boolValue && password.boolValue); }];
-
RAC一些常用的宏
-
RAC(),这个宏的完整写法是:
RAC(TARGET,[KEYPATH,NIL_VALUE])
使用方法:
RAC(label,text) = self.textField.rac_textSignal; RAC(label,text,@"如果是nil就显示这里的文字") = self.textField.rac_textSignal;
最常用的宏
RAC()
一般出现在等号左边,右边是一个RACSignal
,作用是将一个对象(这里是label)的一个属性(这里是label的text)和一个RACSignal
绑定,RACSignal
每产生一个value,都会自动将这个对象的这个属性设置为value。如果value是nil,就设置为第三个参数。(上面的代码的效果是把label的text和textField的文字变动的signal绑定, 每当textField里面的文字有变化,就给label设置新的text。) -
RACObserve(TARGET,KEYPATH)
作用是观察TARGET的KEYPATH属性,产生一个
RACSignal
,相当于KVO。RAC(self.outputLabel, text) = RACObserve(self.model, name);
把label的text和model的name绑定,只要name有变化,label的text也会变化为相应的值。
暂时写这么点吧。 代码点击这里.