什么是RAC?
最近回顾了一下ReactiveCocoa
的方法,也看了一些人的文章,现写篇文章总结一下。
现在这个库最新支持Swift,如果你要是用Cocoapods
的话不指定版本它默认是下载Swift版本,如果依旧想用OC版本就指定一个版本,最好是V2.5版本及以下,否则可能会出现错误。最近我试的是V2.5,可以正常使用。
项目中用Cocoapods
使用:pod "ReactiveCocoa", '~> 2.5'
那什么是RAC勒?想必大家随便谷歌一下就一大片这个概念和文章。
RAC具有函数响应式编程特性,由Matt Diephouse
开源的一个应用于iOS和OS X的新框架。
部分参考来源
为什么要使用RAC?
因为RAC具有高聚合低耦合的思想,使用RAC会让代码更简洁,逻辑更清晰。再结合MVVM架构,让你瞬间爽爆了!
RAC有很多的类,为很多的UI控件都拓展了方法,使得开发大大的简便化,这里就简单的介绍开发过程中用到的方法。
打开应用的初始ViewController,引入ReactiveCocoa的头文件。
#import <ReactiveCocoa/ReactiveCocoa.h>
在控制器中创建一个TextField,SB拖入更方便,然后如下
[self.TextField.rac_textSignal subscribeNext:^(id x){
NSLog(@"x:%@", x);
}];
编译运行,在输入框中输入文字。注意打印信息的输出应该和下面的类似。
2017-11-29 10:26:25.152197+0800 MVVM-Demo[1089:230607] x:a
2017-11-29 10:26:25.159596+0800 MVVM-Demo[1089:230607] x:ah
2017-11-29 10:26:25.385413+0800 MVVM-Demo[1089:230607] x:ahv
2017-11-29 10:26:25.576558+0800 MVVM-Demo[1089:230607] x:ahva
2017-11-29 10:26:25.764013+0800 MVVM-Demo[1089:230607] x:ahvah
2017-11-29 10:26:25.784379+0800 MVVM-Demo[1089:230607] x:ahvahv
2017-11-29 10:26:25.853596+0800 MVVM-Demo[1089:230607] x:ahvahvj
2017-11-29 10:26:25.868552+0800 MVVM-Demo[1089:230607] x:ahvahvja
2017-11-29 10:26:26.002545+0800 MVVM-Demo[1089:230607] x:ahvahvjav
2017-11-29 10:26:26.062553+0800 MVVM-Demo[1089:230607] x:ahvahvjavj
当你看到这些打印信息,你是不是觉得很神奇,都没有监听TextField的方法,它咋就那么牛逼勒。其实RAC内部就帮你做了许多事情。你只要调用相应控件的RAC方法就可以监听到它们的状态了。
那么它是怎么监听怎么做到的勒?
这里要讲几个很重要的RAC类,不涉及RAC原理,内部怎么实现还要大家去阅读源码了。
RACSiganl
1、RACSiganl
信号类,表示将来有数据传递,有数据改变,信号内部接收到数据,就会马上发出数据,外部就可以接收到数据了。就像刚刚上面的例子一样。
2、默认信号都是冷信号,就是这个值改变了它不会触发,只有订阅(调用信号RACSignal的subscribeNext订阅)了这个信号,这个信号才会变为热信号(值一改变就触发),才会触发。
RACSiganl简单使用:
// 1.创建信号
RACSignal *siganl = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
// 每当有订阅者订阅信号,就会调用block。
// 2.发送信号
[subscriber sendNext:@"我是一个信号?"];
// 如果不在发送数据,最好发送信号完成,内部会自动调用[RACDisposable disposable]取消订阅信号。
[subscriber sendCompleted];
return [RACDisposable disposableWithBlock:^{
// 当信号发送完成或者发送错误,就会自动执行这个block,取消订阅信号。
// 执行完Block后,当前信号就不在被订阅了。
NSLog(@"dealloc");
}];
}];
// 3.订阅信号,才会激活信号.
[siganl subscribeNext:^(id x) {
// block调用时刻:每当有信号发出数据,就会调用block.
NSLog(@"数据: %@",x);
}];
打印如下2017-11-29 11:04:34.383754+0800 MVVM-Demo[1185:379135] 数据:我是一个信号? 2017-11-29 11:04:34.383878+0800 MVVM-Demo[1185:379135] dealloc
RACSubscriber
RACSubscriber
订阅者,用于发送信号,这是一个协议,不是一个类,只要遵守这个协议,并且实现方法才能成为订阅者。通过create创建的信号,都有一个订阅者,帮助他发送数据。
RACDisposable
RACDisposable
用于取消订阅或者清理资源,当信号发送完成或者发送错误的时候,就会自动触发它。 当你不想监听某个信号时,可以通过它主动取消订阅信号。
RACSubject
RACSubject
信号提供者,自己可以充当信号,又能发送信号。通常用来代替代理,有了它,就不必要定义代理了。
RACReplaySubject
RACReplaySubject
重复提供信号类,RACSubject的子类。
-
RACReplaySubject
与RACSubject
区别:RACReplaySubject
可以先发送信号,在订阅信号,RACSubject
就不可以。
* 如果一个信号每被订阅一次,就需要把之前的值重复发送一遍,使用重复提供信号类。
* 可以设置capacity数量来限制缓存的value的数量, 即只缓存最新的几个值。
RACReplaySubject使用步骤:
- 创建信号 [RACSubject subject],跟RACSiganl不一样,创建信号时没有block。
- 可以先订阅信号,也可以先发送信号。
- 订阅信号 - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock
- 发送信号 sendNext:(id)value
RACReplaySubject:底层实现和RACSubject不一样。
- 调用sendNext发送信号,把值保存起来,然后遍历刚刚保存的所有订阅者,一个一个调用订阅者的nextBlock。
- 调用subscribeNext订阅信号,遍历保存的所有值,一个一个调用订阅者的nextBlock
- 如果想当一个信号被订阅,就重复播放之前所有值,需要先发送信号,在订阅信号。
- 先保存值,在订阅值。
// 1.创建信号
RACReplaySubject *replaySubject = [RACReplaySubject subject];
// 2.发送信号
[replaySubject sendNext:@1];
[replaySubject sendNext:@2];
// 3.订阅信号
[replaySubject subscribeNext:^(id x) {
NSLog(@"第一个订阅者接收到的数据%@",x);
}];
// 订阅信号
[replaySubject subscribeNext:^(id x) {
NSLog(@"第二个订阅者接收到的数据%@",x);
}];
打印如下:
2017-11-29 11:02:07.468379+0800 MVVM-Demo[1158:370610] 第一个订阅者接收到的数据1
2017-11-29 11:02:07.468477+0800 MVVM-Demo[1158:370610] 第一个订阅者接收到的数据2
2017-11-29 11:02:07.468592+0800 MVVM-Demo[1158:370610] 第二个订阅者接收到的数据1
2017-11-29 11:02:07.468722+0800 MVVM-Demo[1158:370610] 第二个订阅者接收到的数据2
RACSubject使用步骤
- 创建信号 [RACSubject subject],跟RACSiganl不一样,创建信号时没有block。
- 订阅信号 - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock
- 发送信号 sendNext:(id)value
RACSubject:底层实现和RACSignal不一样。
- 调用subscribeNext订阅信号,只是把订阅者保存起来,并且订阅者的nextBlock已经赋值了。
- 调用sendNext发送信号,遍历刚刚保存的所有订阅者,一个一个调用订阅者的nextBlock。
// 1.创建信号
RACSubject *subject = [RACSubject subject];
// 2.订阅信号
[subject subscribeNext:^(id x) {
// 当信号发出新值,就会调用.
NSLog(@"第一个订阅者%@",x);
}];
[subject subscribeNext:^(id x) {
// 当信号发出新值,就会调用.
NSLog(@"第二个订阅者%@",x);
}];
// 3.发送信号
[subject sendNext:@"我是一个信号?"];
RACTuple
RACTuple
元组类, 类似NSArray,用来包装值.
RACSequence
RACSequence
集合类,用于代替NSArray, NSDictionary,可以使用它来快速遍历数组和字典。
RACSequence
和RACTuple
简单使用
NSArray
NSArray *arr = @[@"哈哈",@"呵呵", @"嘿嘿", @"哼哼"];
[arr.rac_sequence.signal subscribeNext:^(id x) {
NSLog(@"x: %@", x);
}];
打印如下
2017-11-29 11:19:27.081935+0800 MVVM-Demo[1267:428560] x: 哈哈
2017-11-29 11:19:27.082227+0800 MVVM-Demo[1267:428560] x: 呵呵
2017-11-29 11:19:27.082350+0800 MVVM-Demo[1267:428560] x: 嘿嘿
2017-11-29 11:19:27.082664+0800 MVVM-Demo[1267:428560] x: 哼哼
原理:
- 通过
arr.rac_sequence
把数据arr转化成集合RACSequence - 通过
arr.rac_sequence.signal
把集合RACSequence转化成了信号
*通过subscribeNext订阅信号,把遍历集合
NSDictionary
// 2.遍历字典,遍历出来的键值对会包装成RACTuple(元组对象)
NSDictionary *dict = @{@"name": @"soliloquy", @"age": @26};
[dict.rac_sequence.signal subscribeNext:^(id x) {
// 解包元组,会把元组的值,按顺序给参数里面的变量赋值
RACTupleUnpack(NSString *key,NSString *value) = x;
NSLog(@"%@ %@",key,value);
}];
打印如下
2017-11-29 11:51:08.027070+0800 MVVM-Demo[1367:471752] name soliloquy
2017-11-29 11:51:08.027526+0800 MVVM-Demo[1367:471752] age 26
字典转模型
NSArray *arr = @[
@{@"name": @"soliloquy", @"age": @26},
@{@"name": @"ptl", @"age": @21},
];
[arr.rac_sequence.signal subscribeNext:^(id x) {
// 运用RAC遍历字典,x:字典
Model *item = [Model modelWithDict:x];
[array addObject:item];
}];
其他用法
NSArray *arr = @[
@{@"name": @"soliloquy", @"age": @26},
@{@"name": @"ptl", @"age": @21},
];
NSArray *ay = [[arr.rac_sequence map:^id(id value) {
return [Persion modelWithDict: value];
}] array];
NSLog(@"ay: %@", ay);
for (Persion *model in ay) {
NSLog(@"%@---%zd", model.name, model.age);
}
打印如下
2017-11-29 12:18:24.024939+0800 MVVM-Demo[1631:553078] ay: (
"<Persion: 0x600000224b00>",
"<Persion: 0x600000224a60>"
)
2017-11-29 12:18:24.025072+0800 MVVM-Demo[1631:553078] soliloquy---26
2017-11-29 12:18:24.025184+0800 MVVM-Demo[1631:553078] ptl---21
RACCommand
RACCommand
RAC 中用于处理事件的类,可以把事件如何处理,事件中的数据如何传递,包装到这个类中,他可以很方便的监控事件的执行过程。
比如:监听按钮点击,网络请求
RACCommand简单使用
一、RACCommand使用步骤:
1.创建命令 initWithSignalBlock:(RACSignal * (^)(id input))signalBlock
2.在signalBlock中,创建RACSignal,并且作为signalBlock的返回值
3.执行命令 - (RACSignal *)execute:(id)input
二、RACCommand使用注意:
1.signalBlock必须要返回一个信号,不能传nil.
2.如果不想要传递信号,直接创建空的信号[RACSignal empty];
3.RACCommand中信号如果数据传递完,必须调用[subscriber sendCompleted],这时命令才会执行完毕,否则永远处于执行中。
4.RACCommand需要被强引用,否则接收不到RACCommand中的信号,因此RACCommand中的信号是延迟发送的。
三、RACCommand设计思想
1.在RAC开发中,通常会把网络请求封装到RACCommand,直接执行某个RACCommand就能发送请求。
2.当RACCommand内部请求到数据的时候,需要把请求的数据传递给外界,这时候就需要通过signalBlock返回的信号传递了。
四、如何拿到RACCommand中返回信号发出的数据。
1.RACCommand有个执行信号源executionSignals,这个是signal of signals(信号的信号),意思是信号发出的数据是信号,不是普通的类型。
2.订阅executionSignals就能拿到RACCommand中返回的信号,然后订阅signalBlock返回的信号,就能获取发出的值。
五、监听当前命令是否正在执行executing
六、使用场景, 监听按钮点击,网络请求
// 1.创建命令 强引用命令,不要被销毁,否则接收不到数据
self.command = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
// 创建空信号,必须返回信号
// return [RACSignal empty];
// 2.创建信号,用来传递数据
return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSArray *arr = @[@"123",@"321", @"132", @"312"];
[subscriber sendNext:arr];
// 注意:数据传递完,最好调用sendCompleted,这时命令才执行完毕。
[subscriber sendCompleted];
return nil;
}];
}];
// 3.订阅RACCommand中的信号
[command.executionSignals subscribeNext:^(id x) {
[x subscribeNext:^(id x) {
NSLog(@"数据为: %@",x);
}];
}];
打印如下
2017-11-29 12:41:50.364091+0800 MVVM-Demo[1844:622694] 数据为: (
123,
321,
132,
312
)
RAC高级用法
// switchToLatest:用于signal of signals,获取signal of signals发出的最新信号,也就是可以直接拿到RACCommand中的信号
[self.command.executionSignals.switchToLatest subscribeNext:^(id x) {
NSLog(@"x: %@",x);
}];
// 4.监听命令是否执行完毕,默认会来一次,可以直接跳过,skip表示跳过第一次信号。
[[command.executing skip:1] subscribeNext:^(id x) {
if ([x boolValue] == YES) {
NSLog(@"正在执行");
}else{
NSLog(@"执行完成");
}
}];
// 5.执行命令
[self.conmmand execute:nil];
RACMulticastConnection
RACMulticastConnection
用于当一个信号被多个订阅时,为了保证创建信号时避免多次调用创建信号中的block造成多次发生数据,可以使用这个该类处理。RACMulticastConnection
通过RACSignal
的-publish或者-muticast:方法创建.
######RACMulticastConnection使用步骤:
- 创建信号
+ (RACSignal *)createSignal:(RACDisposable * (^)(id<RACSubscriber> subscriber))didSubscribe
- 创建连接
RACMulticastConnection *connect = [signal publish];
- 订阅信号,注意:订阅的不在是之前的信号,而是连接的信号。
[connect.signal subscribeNext:nextBlock];
- 连接
[connect connect];
// 1.创建请求信号
RACSignal *signal1 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSLog(@"发送数据");
return nil;
}];
// 2.订阅信号
[signal1 subscribeNext:^(id x) {
NSLog(@"接收数据");
}];
// 2.订阅信号
[signal1 subscribeNext:^(id x) {
NSLog(@"接收数据");
}];
// 3.运行结果,会执行两遍发送请求,也就是每次订阅都会发送一次请求
// RACMulticastConnection:解决重复请求问题
// 1.创建信号
RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
NSLog(@"发送请求");
[subscriber sendNext:@"我是数据源"];
return nil;
}];
// 2.创建连接
RACMulticastConnection *connect = [signal publish];
// 3.订阅信号,
// 注意:订阅信号,也不能激活信号,只是保存订阅者到数组,必须通过连接,当调用连接,就会一次性调用所有订阅者的sendNext:
[connect.signal subscribeNext:^(id x) {
NSLog(@"订阅者一:%@", x);
}];
[connect.signal subscribeNext:^(id x) {
NSLog(@"订阅者二:%@", x);
}];
// 4.连接,激活信号
[connect connect];
打印如下
2017-11-29 13:13:40.735831+0800 MVVM-Demo[2088:721509] 发送数据
2017-11-29 13:13:40.736024+0800 MVVM-Demo[2088:721509] 发送数据
2017-11-29 13:13:40.736550+0800 MVVM-Demo[2088:721509] 发送请求
2017-11-29 13:13:40.736688+0800 MVVM-Demo[2088:721509] 订阅者一:我是数据源
2017-11-29 13:13:40.736777+0800 MVVM-Demo[2088:721509] 订阅者二:我是数据源
可以看出RACSignal
被调了两次,每次订阅一次都会发送请求,这样就会导致多次请求,而使用RACMulticastConnection
就只有在创建信号时调了一次。
RACMulticastConnection底层原理:
- 创建connect
connect.sourceSignal -> RACSignal(原始信号) connect.signal -> RACSubject
- 订阅connect.signal,会调用RACSubject的subscribeNext,创建订阅者,而且把订阅者保存起来,不会执行block。
-
[connect connect]
内部会订阅RACSignal(原始信号),并且订阅者是RACSubject
订阅原始信号,就会调用原始信号中的didSubscribe
didSubscribe,拿到订阅者调用sendNext,其实是调用RACSubject的sendNext
*RACSubject
的sendNext
,会遍历RACSubject所有订阅者发送信号。拿到第二步所有的订阅者,调用他们的nextBlock