RAC(ReactiveCocoa)
简单的说,RAC就是一个第三方库,他可以大大简化你的代码过程。
官方的说,ReactiveCocoa(其简称为RAC)是由GitHub开源的一个应用于iOS和OS X开发的新框架。RAC具有函数式编程和响应式编程的特性。
我自己在练习了一些demo时,逐渐感受到到RAC的强大,信号随数据流的变化给订阅者发送信号,发送信号就是你要做得事件。这就是响应式编程,根据数据流表达应用的功能。慢慢的体会到信号的过人之处,之前要让自己编写这样的代码会需要多次,甚至更多次调用代理,方法的地方现在都只需要我们订阅信号,信号发送就可以了。
RAC常用的类
1. Stream - 信号流值 - RACStream类
表示一个基本单元可以为任意值,其值会随着事件的变化而变化,可以在其上进行一些复杂的操作运算(map,filter,skip,take等.)此类不会被经常使用, 多情况下表现为signal和sequences(RACSignal 和RACSequence继承于RACStream类)
2. Signals - 信号 - RACSignal类
有订阅者监听时信号才会发信息, Signals会向那个订阅者发送0或多个载有数值的”next”事件,后面跟着一个”complete”事件或一个”error”事件。
Signals会发送三种不同信号给Subscriber
next:是可以为nil的新值, RACStream方法只能在这个值上进行操作运算。
error:表示在Signals完成之前发生了错误,值不会在RACStream类中存储。
completed:表示Signals成功的完成,值不会在RACStream类中存储。
3. Subscriber - 订阅者 - RACSubscriber协议
表示能够接收信号的对象,订阅信号才会激活信号,实现RACSubscriber协议的对象都可以为订阅者。
可以通过- (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock 方法创建Subscriber。
4. Subjects - 手动控制信号 - RACSubject
表示可以手动控制信号,
处理流程:创建信号-订阅信号-发送信号
// 1.创建信号
RACSubject *subject = [RACSubject subject];
// 2.订阅信号 First
[subject subscribeNext:^(id x) {
// block调用时刻:当信号发出新值,就会调用.
NSLog(@"FirstSubscribeNext%@",x);
}];
// 2.订阅信号 Second
[subject subscribeNext:^(id x) {
// block调用时刻:当信号发出新值,就会调用.
NSLog(@"SecondSubscribeNext%@",x);
}];
// 3.发送信号
[subject sendNext:@"1"];
[subject sendNext:@"2"];
也是RAC代码与非RAC代码的Bridge 所以非常有用,此类继承于RACSignal类。
5. ReplaySubject - 手动回放控制信号 - RACReplaySubject
表示可以手动控制信号,底层实现和RACSubject不一样,它会先把值保存起来,然后遍历刚刚保存的所有订阅者,一个一个调用订阅者的nextBlock然后调用subscribeNext订阅信号,遍历保存的所有值,一个一个调用订阅者的nextBlock。
可以有以下两种处理流程:
处理流程 1:创建信号-订阅信号-发送信号(和Subjects一样)
处理流程 2:创建信号-发送信号-订阅信号
// 1.创建信号
RACReplaySubject *replaySubject = [RACReplaySubject subject];
// 2.发送信号
[replaySubject sendNext:@"1、hello"];
[replaySubject sendNext:@"2、你好"];
// 3.订阅信号 First
[replaySubject subscribeNext:^(id _Nullable x) {
NSLog(@"FirstSubscribeNext:%@",x);
}];
// 3.订阅信号 Second
[replaySubject subscribeNext:^(id x) {
NSLog(@"SecondSubscribeNext%@",x);
}];
6. Command- 命令信号 - RACCommand
表示订阅响应Action信号,通常由UI来出发,比如一个Button当控件被触发时会被自动禁用掉。
button.rac_command = [[RACCommand alloc]initWithSignalBlock:^RACSignal * _Nonnull(id _Nullable input) {
NSLog(@"按钮被点击");
NextViewViewController *nextVC = [[NextViewViewController alloc] init];
[self presentViewController:nextVC animated:YES completion:nil];
return [RACSignal empty];
}];
7. Sequences- 集合 - RACSequence
表示一个不可变的序列值且不能包含空值,使用-rac_sequence.signal来获取Signal。
RACSignal *signal = [@"A B C D E F G H I" componentsSeparatedByString:@" "].rac_sequence.signal;
[signal subscribeNext:^(id _Nullable x) {
NSLog(@"%@",x);
}];
2017-11-21 17:44:25.213821+0800 study[9679:354266] A
2017-11-21 17:44:25.214216+0800 study[9679:354266] B
2017-11-21 17:44:25.214895+0800 study[9679:354266] C
2017-11-21 17:44:25.215244+0800 study[9679:354266] D
2017-11-21 17:44:25.215894+0800 study[9679:354266] E
2017-11-21 17:44:25.216166+0800 study[9679:354266] F
2017-11-21 17:44:25.216604+0800 study[9679:354266] G
2017-11-21 17:44:25.217259+0800 study[9679:354266] H
2017-11-21 17:44:25.217981+0800 study[9679:354266] I
一些注意的点
用RAC写代理是有局限的,它只能实现返回值为void的代理方法
8.ReactiveCocoa开发中常见用法
1、 代替代理:
rac_signalForSelector:用于替代代理。
2、 代替KVO :
rac_valuesAndChangesForKeyPath:用于监听某个对象的属性改变。
3、 监听事件:
rac_signalForControlEvents:用于监听某个事件。
4、 代替通知:
rac_addObserverForName:用于监听某个通知。
5、 监听文本框文字改变:
rac_textSignal:只要文本框发出改变就会发出这个信号。
6、 处理当界面有多次请求时,需要都获取到数据时,才能展示界面
rac_liftSelector:withSignalsFromArray:Signals:当传入的Signals(信号数组),每一个signal都至少sendNext过一次,就会去触发第一个selector参数的方法。
使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
比如:
RAC上设置button的controlEvents
//点击button使得选中状态改变
[[button rac_signalForControlEvents:UIControlEventTouchUpInside]
subscribeNext:^(UIButton * button) {
button.selected = !button.selected;
}];
9.ReactiveCocoa常见宏
1、 RAC(TARGET, [KEYPATH, [NIL_VALUE]]):用于给某个对象的某个属性绑定。
// 只要文本框文字改变,就会修改label的文字
RAC(self.labelView,text) = _textField.rac_textSignal;
2、RACObserve(self, name):监听某个对象的某个属性,返回的是信号。
[RACObserve(self.view, center) subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
3、 @weakify(Obj)和@strongify(Obj),一般两个都是配套使用,在主头文件(ReactiveCocoa.h)中并没有导入,需要自己手动导入,RACEXTScope.h才可以使用。但是每次导入都非常麻烦,只需要在主头文件自己导入就好了。
4、RACTuplePack:把数据包装成RACTuple(元组类
// 把参数中的数据包装成元组
RACTuple *tuple = RACTuplePack(@10,@20);
5、RACTupleUnpack:把RACTuple(元组类)解包成对应的数据。
// 把参数中的数据包装成元组
RACTuple *tuple = RACTuplePack(@"xmg",@20);
// 解包元组,会把元组的值,按顺序给参数里面的变量赋值
// name = @"xmg" age = @20
RACTupleUnpack(NSString *name,NSNumber *age) = tuple;
参考:
http://www.jianshu.com/p/e99cb4310482