什么是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的子类。

  • 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,可以使用它来快速遍历数组和字典。

  • 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 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

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<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订阅原始信号,就会调用原始信号中的didSubscribedidSubscribe,拿到订阅者调用sendNext,其实是调用RACSubject的sendNext *RACSubjectsendNext,会遍历RACSubject所有订阅者发送信号。
    拿到第二步所有的订阅者,调用他们的nextBlock