RAC(ReactiveCocoa)使用方法(一)

RAC(ReactiveCocoa)使用方法(一) RAC(ReactiveCocoa)使用方法(二) 什么是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 在控制器中创建一个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 // 每当有订阅者订阅信号,就会调用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: ( " * #####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 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 subscriber))didSubscribe 创建连接 RACMulticastConnection *connect = [signal publish]; 订阅信号,注意:订阅的不在是之前的信号,而是连接的信号。 [connect.signal subscribeNext:nextBlock]; 连接 [connect connect]; // 1.创建请求信号 RACSignal *signal1 = [RACSignal createSignal:^RACDisposable *(id 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 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所有订阅者发送信号。 拿到第二步所有的订阅者,调用他们的nextBlockhttp://www.cnblogs.com/soliloquy/p/7920551.html
50000+
5万行代码练就真实本领
17年
创办于2008年老牌培训机构
1000+
合作企业
98%
就业率

联系我们

电话咨询

0532-85025005

扫码添加微信