美文网首页IOS知识积累
ReactiveCocoa学习笔记整理(二)

ReactiveCocoa学习笔记整理(二)

作者: 爱敲代码的果果 | 来源:发表于2017-11-21 19:01 被阅读85次

    此文章是ReactiveCocoa学习笔记的第二篇,未阅读第一篇的童鞋,请先查阅第一篇ReactiveCocoa学习笔记整理(一) 。好的,废话不多说,上一篇中我们简单的了解了RAC的概念,编程思想以及RACSignal这个基类的简单应用。我们接着上一篇的内容继续探寻RAC学习之路。

    三. RACSubject基础知识点

    首先,我们介绍另一个使用非常频繁的类,RACSubject,我们将会从它的概念,使用步骤,底层实现入手,慢慢剖析,然后通过几个简单的小例子应用一下下。

    1. RACSubject基本概念以及使用步骤

    RACSubject被称为信号提供者,它既可以自己充当信号,又能够发送信号。最常用的使用场景就是用来代替代理,有了它,就没必要定义代理了。接下来,我们看一下RACSubject的具体使用步骤:

    1. 创建信号 [RACSubject subject],跟RACSiganl不一样,创建信号时没有block。
    2. 订阅信号 - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock
    3. 发送信号 sendNext:(id)value

    2. RACSubject底层实现以及简单应用

    RACSubject的底层实现和RACSignal不一样,简单的介绍一下RACSubject的实现步骤。

    1. 调用subscribeNext订阅信号,只是把订阅者保存起来,并且订阅者的nextBlock已经赋值了
    2. 调用sendNext发送信号,遍历刚刚保存的所有订阅者,一个一个调用订阅者的nextBlock
      在这篇文章就不展开描述了,有兴趣的童鞋可以查阅ReactiveCocoa深入理解 ,里面有关于底层实现的详细剖析。这里需要对持有订阅者解释一下,由于RACSubject是热信号, 为了保证未来有事件发生的时候, 订阅者可以收到信息, 所以需要持有订阅者。好了,我们用简单的小实例探究一下,请看如下代码:
    RACSubject *subject=[RACSubject subject];
        
        [subject subscribeNext:^(id x) {
            NSLog(@"订阅者1的值%@",x);
        }];
        
        [subject sendNext:@"wujunyang"];
        
        [subject subscribeNext:^(id x) {
            NSLog(@"订阅者2的值%@",x);
        }];
        
        [subject sendNext:@"cnblogs"];
        
    //    输出:
    //    订阅者1的值wujunyang
    //    订阅者1的值cnblogs
    //    订阅者2的值cnblogs
    

    通过以后的代码输出,我们可以很明确的得到结论:RACSubject 发送过了sendNext, 下面再去监听它是没有效果,即RACSubject是热信号的本质。
    上面也提到了,RACSubject最常用的使用场景就是用来代替代理,接下来我们写一个小的需求来看一下是如何实现的代替代理。目前,我们有这样一个需求:

    • 给当前控制器添加一个按钮,modal到另一个控制器界面
    • 另一个控制器view中有个按钮,点击按钮,通知当前控制器
      我们用RACSubject可以这样来实现:
     
        @interface TwoViewController : UIViewController
        
        @property (nonatomic, strong) RACSubject *delegateSignal;
        
        @end
        
    
        @implementation TwoViewController
        - (IBAction)notice:(id)sender {
            // 通知第一个控制器,告诉它,按钮被点了
            
            // 通知代理
            // 判断代理信号是否有值
            if (self.delegateSignal) {
                // 有值,才需要通知
                [self.delegateSignal sendNext:nil];
            }
        }
        @end
        
        @implementation OneViewController
        - (IBAction)btnClick:(id)sender {
            
            // 创建第二个控制器
            TwoViewController *twoVc = [[TwoViewController alloc] init];
            
            // 设置代理信号
            twoVc.delegateSignal = [RACSubject subject];
            
            // 订阅代理信号
            [twoVc.delegateSignal subscribeNext:^(id x) {
                
                NSLog(@"点击了通知按钮");
            }];
            
            // 跳转到第二个控制器
            [self presentViewController:twoVc animated:YES completion:nil];
            
        }
        @end
    

    在上述代码中,大致分为三个步骤:

    1. 在第二个控制器.h,添加一个RACSubject代替代理
    2. 监听第二个控制器按钮点击
    3. 在第一个控制器中,监听跳转按钮,给第二个控制器的代理信号赋值,并且监听
      你看,用RACSubject代替代理就是这么简单滴就实现啦。

    3. RACReplaySubject简单介绍以及应用

    既然讲到了RACSubject,那么久一起把它的子类RACReplaySubject顺便说了吧... RACReplaySubject是重复提供信号类,它的使用场景一般有以下两个:

    • 如果一个信号每被订阅一次,就需要把之前的值重复发送一遍,使用重复提供信号类
    • 可以设置 capacity 数量来限制缓存的 value 的数量,即只缓充最新的几个值
      RACReplaySubject的创建方法也很简单明了,首先创建RACReplaySubject,然后订阅信号,最后发送信号。而它的工作流程可以大致分为以下的三步:
    1. 订阅信号时,内部保存了订阅者,和订阅者响应block
    2. 当发送信号时,遍历订阅者,调用订阅者的nextBlock
    3. 发送的信号会保存起来,当订阅者订阅信号时,会将之前保存的信号,一个一个遍历
      最后,我们通过简单的小例子来看一下它的使用:
        RACReplaySubject *replaySubject = [RACReplaySubject subject];
        [replaySubject subscribeNext:^(id x) {
            NSLog(@"1 %@,type:%@",x,NSStringFromClass(object_getClass(x)));
        }];
        [replaySubject subscribeNext:^(id x) {
            NSLog(@"1 %@,type:%@",x,NSStringFromClass(object_getClass(x)));
        }];
        [replaySubject sendNext:@1];
        
        [replaySubject subscribeNext:^(id x) {
            NSLog(@"3 %@,type:%@",x,NSStringFromClass(object_getClass(x)));
        }];
    
        //输出
    //    1 1,type:__NSCFNumber
    //    1 1,type:__NSCFNumber
    //    3 1,type:__NSCFNumber
    

    通过上面的代码,我们可以看出RACReplaySubject与RACSubject的区别,RACSubject必须要先订阅信号之后才能发送信号,而RACReplaySubject可以先发送信号后订阅。

    四. RACSequence以及RACTuple的基本简介跟使用

    啊哈哈,在本章节我们将隆重的介绍RACSequence,它是RAC中的集合类,可以用于代替NSArray,NSDictionary,可以使用它来快速遍历数组和字典。我们就对数组的操作简单的总结一下它的流程:

    1. 把数组转换成集合RACSequence numbers.rac_sequence
    2. 把集合RACSequence转换RACSignal信号类 numbers.rac_sequence.signal
    3. 订阅信号,激活信号,会自动把集合中的所有值,遍历出来
        NSArray *arr = @[@1,@2,@3,@4,@5,@6];
        [arr.rac_sequence.signal subscribeNext:^(id x) {
            
            NSLog(@"当前的值x:%@",x);
        }];
        //输出
    //    当前的值x:1
    //    当前的值x:2
    //    当前的值x:3
    //    当前的值x:4
    //    当前的值x:5
    //    当前的值x:6
        
        
        NSDictionary *dict = [NSDictionary dictionaryWithObjectsAndKeys:@"jtd",@"name",@"man",@"sex",@"jx",@"jg", nil];
        [dict.rac_sequence.signal subscribeNext:^(id x) {
            RACTupleUnpack(NSString *key,NSString *value) = x;
            
            NSLog(@"key:%@,value:%@",key,value);
        }];
        //输出:
    //    key:name,value:jtd
    //    key:sex,value:man
    //    key:jg,value:jx
    

    如以上代码所示,就如此方便快捷的实现了对数组以及字典的遍历操作。同样的也可以对转化之后的数组进行诸如map,filter,reduce,skip,take,contact..等等操作,接下来,我们通过代码来详细看一下:

        NSArray *array=@[@(2),@(5),@(7),@(15)];
        RACSequence *sequence = [array rac_sequence];
        
        id mapData = [sequence map:^id(id value) {
            return @([value integerValue] * 2);
        }];
        NSLog(@"序列Map之后的数据:%@",[mapData array]);
        
        id filterData = [sequence filter:^BOOL(id value) {
            return [value integerValue]%2 == 0;
        }];
        NSLog(@"序列Filter之后的数据:%@",[filterData array]);
        
        id reduceData = [sequence foldLeftWithStart:@"" reduce:^id(id accumulator, id value) {
            return [accumulator stringByAppendingString:[value stringValue]];
        }];
        NSLog(@"序列Left-Reduce之后的数据:%@",reduceData);
        
        id rightReduceData = [sequence foldRightWithStart:@"" reduce:^id(id first, RACSequence *rest) {
            return [NSString stringWithFormat:@"%@%@", rest.head, first];
        }];
        NSLog(@"序列Right-Reduce之后的数据:%@",rightReduceData);
        
        id skipData = [sequence skip:1];
        NSLog(@"序列Skip之后的数据:%@",[skipData array]);
        
        
        id takeData = [sequence take:2];
        NSLog(@"序列Take之后的数据:%@",[takeData array]);
        
        id takeUntilData = [sequence takeUntilBlock:^BOOL(id x) {
            return [x integerValue] == 7;
        }];
        NSLog(@"序列TakeUntil之后的数据:%@",[takeUntilData array]);
        
        NSArray *nextArr = @[@"A",@"B",@"C"];
        RACSequence *nextSequence = [nextArr rac_sequence];
        id contactData = [sequence concat:nextSequence];
        NSLog(@"FlyElephant序列Contact之后的数据:%@",[contactData array]);
    
        //输出
    //    序列Map之后的数据:(
    //                                           4,
    //                                           10,
    //                                           14,
    //                                           30
    //                                           )
    //    序列Filter之后的数据:(
    //                                                                      2
    //                                                                      )
    //    序列Left-Reduce之后的数据:25715
    //    序列Right-Reduce之后的数据:15752
    //    序列Skip之后的数据:(
    //                                                                    5,
    //                                                                    7,
    //                                                                    15
    //                                                                    )
    //    序列Take之后的数据:(
    //                                                                    2,
    //                                                                    5
    //                                                                    )
    //    序列TakeUntil之后的数据:(
    //                                                                         2,
    //                                                                         5
    //                                                                         )
    //    FlyElephant序列Contact之后的数据:(
    //                                                                                  2,
    //                                                                                  5,
    //                                                                                  7,
    //                                                                                  15,
    //                                                                                  A,
    //                                                                                  B,
    //                                                                                  C
    //                                                                                  )
    

    然后,我们再来看一下RACTuple这个类,RACTuple是ReactiveCocoa的元组类,首先上图看一下他的定义。

    @interface RACTuple : NSObject <NSCoding, NSCopying, NSFastEnumeration>
    
    @property (nonatomic, readonly) NSUInteger count;
    
    @property (nonatomic, readonly) id first;
    @property (nonatomic, readonly) id second;
    @property (nonatomic, readonly) id third;
    @property (nonatomic, readonly) id fourth;
    @property (nonatomic, readonly) id fifth;
    @property (nonatomic, readonly) id last;
    @property (nonatomic, strong) NSArray *backingArray;
    
    @property (nonatomic, copy, readonly) RACSequence *rac_sequence; // 这个是专门为sequence提供的一个扩展
    
    @end
    

    可以看到,RACTuple的定义看上去很简单,底层实质就是一个NSArray,只不过封装了一些方法。RACTuple继承了NSCoding, NSCopying, NSFastEnumeration这三个协议,具体的协议就不做过多的解读了,本文本着 "如何使用" 来进行基础讲解,况且RACTuple的方法也不多,总共就6个方法,3个类方法,3个实例方法。RACTuple有如下的优点:

    • 可用下标访问元素 (实现了objectAtIndexedSubscript:方法)
    • 可用for in枚举(遵循NSFastEnumeration协议)
    • 可跟NSArray互相转换
    • 可转换为RACSequence
    • 可把NSNull.null转为RACTupleNil.tupleNil
      我们通过简单的实例来看一下用法,请看如下代码:
        //普通创建
        RACTuple *tuple1 = [RACTuple tupleWithObjects:@1, @2, @3, nil];
        RACTuple *tuple2 = [RACTuple tupleWithObjectsFromArray:@[@1, @2, @3]];
        RACTuple *tuple3 = [[RACTuple alloc] init];
        
        //宏创建
        RACTuple *tuple4 = RACTuplePack(@1, @2, @3, @4);
        
        //解包(等号前面是参数定义,后面是已存在的Tuple,参数个数需要跟Tuple元素相同)
        RACTupleUnpack(NSNumber * value1, NSNumber * value2, NSNumber * value3, NSNumber * value4) = tuple4;
        NSLog(@"%@ %@ %@ %@", value1, value2, value3, value4);
        
        //元素访问方式
        NSLog(@"%@", [tuple4 objectAtIndex:1]);
        NSLog(@"%@", tuple4[1]);
        
        //输出
        //1 2 3 4
        //2
        //2
    

    好了,今天的文章就到此吧,接下来的文章,我们将要探讨一下RACCommand以及简单的UIKit的实际运用,下篇文章见。

    相关文章

      网友评论

        本文标题:ReactiveCocoa学习笔记整理(二)

        本文链接:https://www.haomeiwen.com/subject/rkiwvxtx.html