ReactiveCocoa学习笔记整理(二)

此文章是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的实际运用,下篇文章见。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,425评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,058评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,186评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,848评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,249评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,554评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,830评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,536评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,239评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,505评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,004评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,346评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,999评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,060评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,821评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,574评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,480评论 2 267

推荐阅读更多精彩内容