博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
RAC(ReactiveCocoa)使用方法(一)
阅读量:7048 次
发布时间:2019-06-28

本文共 11125 字,大约阅读时间需要 37 分钟。

什么是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:a2017-11-29 10:26:25.159596+0800 MVVM-Demo[1089:230607] x:ah2017-11-29 10:26:25.385413+0800 MVVM-Demo[1089:230607] x:ahv2017-11-29 10:26:25.576558+0800 MVVM-Demo[1089:230607] x:ahva2017-11-29 10:26:25.764013+0800 MVVM-Demo[1089:230607] x:ahvah2017-11-29 10:26:25.784379+0800 MVVM-Demo[1089:230607] x:ahvahv2017-11-29 10:26:25.853596+0800 MVVM-Demo[1089:230607] x:ahvahvj2017-11-29 10:26:25.868552+0800 MVVM-Demo[1089:230607] x:ahvahvja2017-11-29 10:26:26.002545+0800 MVVM-Demo[1089:230607] x:ahvahvjav2017-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
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的子类。

  • RACReplaySubjectRACSubject区别: 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] 第一个订阅者接收到的数据12017-11-29 11:02:07.468477+0800 MVVM-Demo[1158:370610] 第一个订阅者接收到的数据22017-11-29 11:02:07.468592+0800 MVVM-Demo[1158:370610] 第二个订阅者接收到的数据12017-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,可以使用它来快速遍历数组和字典。

  • RACSequenceRACTuple简单使用

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 soliloquy2017-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: (    "
", "
")2017-11-29 12:18:24.025072+0800 MVVM-Demo[1631:553078] soliloquy---262017-11-29 12:18:24.025184+0800 MVVM-Demo[1631:553078] ptl---21复制代码
  • #####RACCommand

RACCommandRAC 中用于处理事件的类,可以把事件如何处理,事件中的数据如何传递,包装到这个类中,他可以很方便的监控事件的执行过程。 比如:监听按钮点击,网络请求

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<RACSubscriber> 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 *RACSubjectsendNext,会遍历RACSubject所有订阅者发送信号。 拿到第二步所有的订阅者,调用他们的nextBlock

转载地址:http://kbkol.baihongyu.com/

你可能感兴趣的文章
一步一步写一个简单通用的makefile(二)
查看>>
sunspot使用
查看>>
Zombie.js Insanely fast, headless full-stack testing using Node.js
查看>>
POJ2406-Power Strings(kmp循环节)
查看>>
BCM路由全智能固件升级软件tftp,一键刷路由及常用固件下载
查看>>
个人认识:直接断电和发送复位信号给主板有啥区别?
查看>>
测试体会:WAYOS新架构(即二代QOS)的新功能解释
查看>>
UVA 10169 Urn-ball Probabilities !
查看>>
每日一例,练就编程高手
查看>>
no argument specified with option "/LIBPATH:"错误的解决【转载】
查看>>
初涉c#设计模式-Factory Pattern
查看>>
android 广播复杂参数
查看>>
EmbossMaskFilter BlurMaskFilter
查看>>
android中XML文件解析遇到“not well-formed (invalid token)”解决办法
查看>>
JRuby——Java和Ruby的强强联合
查看>>
ipcs和ipcrm用法简介
查看>>
[Go 笔记]关于 Panic和 Recover
查看>>
关于HP Diagnostics
查看>>
Oracle中的二进制、八进制、十进制、十六进制相互转换函数
查看>>
关于empty函数的输出
查看>>