RAC(ReactiveCocoa)使用方法

Posted ptlCoder的博客园

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了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 <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

    // 每当有订阅者订阅信号,就会调用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] 第一个订阅者接收到的数据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<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
    *RACSubjectsendNext,会遍历RACSubject所有订阅者发送信号。
    拿到第二步所有的订阅者,调用他们的nextBlock

以上是关于RAC(ReactiveCocoa)使用方法的主要内容,如果未能解决你的问题,请参考以下文章

IOS响应式编程框架ReactiveCocoa(RAC)使用示例-备

RAC(ReactiveCocoa)使用方法

iOS-RAC从新手开始

RAC(ReactiveCocoa)

最快让你上手ReactiveCocoa之基础篇(简称RAC)

iOS - Rac(ReactiveCocoa)