ReactiveCocoa 初探

最近每周末都会看叶孤城的直播,对于我们iOS开发者来说,确实是一个福利,很感谢他们的分享精神,收获到的一些东西特此记录下。

12月19号 ReactiveCocoa

昨天听了DeveloperLx的视频之后,对ReactiveCocoa有了个初步的认识下,暂时可能不会用到,但是了解还是必须的。ReactiveCocoa(简称为RAC),是由Github开源的一个应用于iOS和OS开发的新框架,兼具函数式编程响应式编程 的特性,可以很好的用于消息传递、回调机制复杂等问题,使之清晰化,条理化。

ReactiveCocoa结合了一些编程模式:

  • 函数式编程:利用高阶函数,即将函数作为其它函数的参数。
  • 响应式编程:关注于数据流及变化的传播。

基于以上两点,ReactiveCocoa被当成是函数响应编程(Functional Reactive Programming, FRP)框架。

一、导入ReactiveCocoa 框架

我们可以直接进入到ReactiveCocoa的github了解下,通常我们用CocoaPods就OK啦

pod 'ReactiveCocoa'

很多情况下,直接导入就可以了,但是这里会报这个错

屏幕快照 2015-12-20 下午4.58.42.png

需要在Podfile加上use_frameworks!,重新pod install 才能导入成功

use_frameworks!
pod 'ReactiveCocoa'

但是我使用Xcode7.2的时候,还是出现下面这个问题

Box.swift: error: 'Printable' has been renamed to 'CustomStringConvertible'
Box.swift: error: 'toString' has been renamed to 'String'
Box/MutableBox.swift: error: 'Printable' has been renamed to 'CustomStringConvertible'
MutableBox.swift: error: 'toString' has been renamed to 'String'

大致原因是 这个默认的分支中 swift 不支持swift2.0版的,然后我就视图转换成~> 4.0.4-alpha-1就OK了

use_frameworks!
pod 'ReactiveCocoa','~> 4.0.4-alpha-1'

If you would prefer to use CocoaPods, there are some unofficial pod specs) that have been generously contributed by third parties

二、基本使用

#import <ReactiveCocoa/ReactiveCocoa.h> // 导入头文件
2-1、监听文本框使用
- (void)learnRACWithTextFiled
{
//    // 直接监听 textFiled的改变
//    [[self.testTextField rac_signalForControlEvents:UIControlEventEditingChanged] subscribeNext:^(id x){
//        
//        NSLog(@"%@", x);
//    
//    
//    }];
    
    // 或者
    [self.testTextField.rac_textSignal subscribeNext:^(NSString * textString) {
        
        NSLog(@"%@", textString);
    }];
    
}
// 打印出其textFiled中的文本信息来
2-3、 监听Button事件
- (void)learnRACWithButton
{
    [[self.testButton rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
        
        NSLog(@"按钮被点击了");
    }];
    
}
2-3、手势
- (void)learnRACWithGesture
{
    UITapGestureRecognizer * tap = [[UITapGestureRecognizer alloc]init];
    [self.view addGestureRecognizer:tap];
    
    [[tap rac_gestureSignal] subscribeNext:^(UITapGestureRecognizer * tap) {
        
        // 点击可以
         [[[UIApplication sharedApplication] keyWindow] endEditing:YES];
    }];

}
2-4、通知
-  (void)learnRACWithNSNotificationCenter
{
    // 通知可以不移除
    [[[NSNotificationCenter defaultCenter]
      rac_addObserverForName:UIKeyboardWillShowNotification object:nil]
               subscribeNext:^(NSNotification * notification) {
        
        NSLog(@"show");
    
    }];
}
2-5、定时器
-  (void)learnRACWithNSTimer
{
    NSLog(@"begin");
    //    1. 延迟某个时间后再做某件事
    [[RACScheduler mainThreadScheduler]afterDelay:2.0f schedule:^{
        
        NSLog(@"2秒之后发生的事情");
       
    }];
    
//    2. 每个一定长度时间做一件事
    [[RACSignal interval:4 onScheduler:[RACScheduler mainThreadScheduler]]subscribeNext:^(NSDate * date) {
        
        NSLog(@"每隔几秒发生的事情");
    }];
    /*
     2015-12-21 13:22:23.209 ReactiveCocoaLearn[78775:4675706] begin
     2015-12-21 13:22:25.409 ReactiveCocoaLearn[78775:4675706] 2秒之后发生的事情
     2015-12-21 13:22:27.213 ReactiveCocoaLearn[78775:4675706] 每隔几秒发生的事情
     2015-12-21 13:22:31.211 ReactiveCocoaLearn[78775:4675706] 每隔几秒发生的事情
     
     */

}
2-6、代理

但是有局限,只能取代没有返回值的代理方法

- (void)learnRACWithProtocol
{
    UIAlertView * alertView = [[UIAlertView alloc]initWithTitle:@"RAC中Protocol"
                                                        message:@"UIAlertView"
                                                       delegate:self
                                              cancelButtonTitle:@"Cancel"
                                              otherButtonTitles:@"OK", nil];
    [alertView show];
    [[self rac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)] subscribeNext:^(RACTuple * tuple) {
        
        //可以多尝试下RACTuple里的属性
        NSLog(@"tuple.second == %@",tuple.second);
        if([tuple.second isEqualToNumber:@0])
        {
            NSLog(@"cancel");
        }
        if([tuple.second isEqualToNumber:@1])
        {
            NSLog(@"ok");
        }
        
    }];
    
    //  更简单的方式:
//        [[alertView rac_buttonClickedSignal]subscribeNext:^(id x) {
//            //可以多尝试下RACTuple里的属性
//            NSLog(@"%@",x);
//            if([x isEqualToNumber:@0])
//            {
//                NSLog(@"Cancel");
//            }
//            if([x isEqualToNumber:@1])
//            {
//                NSLog(@"Ok");
//            }
//            
//        }];
    
}
2-7、KVO
[RACObserve(self.testScrollerView, contentOffset) subscribeNext:^(id x) {
    
     NSLog(@"Offset=%@",x);
}];

/*
 2015-12-21 15:06:23.689 ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0, 0}
 2015-12-21 15:06:23.689 ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0, 0}
 2015-12-21 15:06:23.711 ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0, -1}
 2015-12-21 15:06:23.790 ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0, -1.5}
 2015-12-21 15:06:23.870 ReactiveCocoaLearn[81607:4756461] Offset=NSPoint: {0, -2}
 
 */

或是

   [[self.greenView rac_valuesAndChangesForKeyPath:@"center"
                                        options:NSKeyValueObservingOptionNew observer:nil]
 subscribeNext:^(id x) {
    
    NSLog(@"center===%@",x);
    
}];

/*
 center===<RACTuple: 0x7fc7205138c0> (
 "NSPoint: {187.5, 333.5}",
 {
 kind = 1;
 new = "NSPoint: {187.5, 333.5}";
 }
 )
 */

以上是一些RAC的基本用法,熟练这几个以后,我们很多场景都能运用自如,而且会发现RAC真的很方便。

三、RACSignal使用

其实在RAC中最核心的类RACSiganl,搞定这个类就能用ReactiveCocoa开发了。

RACSiganl:信号类,一般表示将来有数据传递,只要有数据改变,信号内部接收到数据,就会马上发出数据。

创建信号 & 激活信号 & 废弃信号

// 1.创建信号 + (RACSignal *)createSignal:(RACDisposable * (^)(id<RACSubscriber> subscriber))didSubscribe
// 2.订阅信号,才会激活信号. - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock
// 3.发送信号 - (void)sendNext:(id)value
// 4.废弃信号  RACDisposable  

  // 创建信号
 RACSignal *siganl = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    
    // block调用时刻:每当有订阅者订阅信号,就会调用block。
    
    // 发送信号
    [subscriber sendNext:@1];
    
    // 如果不在发送数据,最好发送信号完成,内部会自动调用[RACDisposable disposable]取消订阅信号。
    [subscriber sendCompleted];
    
    return [RACDisposable disposableWithBlock:^{
        // 销毁信号
        // block调用时刻:当信号发送完成或者发送错误,就会自动执行这个block,取消订阅信号。
        // 执行完Block后,当前信号就不在被订阅了。
        
        NSLog(@"信号销毁");
        
      }];
  }];

// 订阅信号,才会激活信号.
   [siganl subscribeNext:^(id x) {
        NSLog(@"接到数据x=%@",x);
    }];
/*
 2015-12-21 15:47:18.335 ReactiveCocoaLearn[82287:4789675] 接到数据x=1
 2015-12-21 15:47:18.335 ReactiveCocoaLearn[82287:4789675] 信号销毁
 */
信号的处理
3-1、map
[[self.testTextField.rac_textSignal map:^id(NSString *textStr){

    return @(textStr.length);
}] subscribeNext:^(id x){
    
    NSLog(@"x==%@",x);
}];
// 映射
3-2、filter
[[[self.testTextField.rac_textSignal map:^id(NSString *textStr){

    return @(textStr.length);
}] filter:^BOOL(NSNumber * value){
    
    return value.integerValue > 2;
    
}] subscribeNext:^(id x){
    
    NSLog(@"x==%@",x);
}];

过滤掉一部分

3-3、delay
RACSignal *siganl = [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    
    NSLog(@"realySendSignal");
    [subscriber sendNext:@1];
    [subscriber sendCompleted];
    
    return [RACDisposable disposableWithBlock:^{
        NSLog(@"discard Signal");
    }];
}] delay:3];
NSLog(@"SubscriSiganl");
[siganl subscribeNext:^(id x) {
    
    NSLog(@"recevieSiganl=%@",x);
}];

// 延迟3秒才接收数据
/*
 2015-12-21 16:33:05.326 ReactiveCocoaLearn[83488:4831881] 开始预订信号
 2015-12-21 16:33:05.327 ReactiveCocoaLearn[83488:4831881] 真正发送信号
 2015-12-21 16:33:05.328 ReactiveCocoaLearn[83488:4831881] 销毁信号
 2015-12-21 16:33:08.621 ReactiveCocoaLearn[83488:4831881] 接收信号=1
 */

注意打印的时间,发送信号,订阅信号 的时间,再次了解下整个流程。

3-4、startWith
RACSignal *siganl = [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    

    [subscriber sendNext:@"one"];
    [subscriber sendCompleted];
    
    return [RACDisposable disposableWithBlock:^{
       
    }];
}] startWith:@"two"];

[siganl subscribeNext:^(id x) {
    
    NSLog(@"接收信号=%@",x);
}];

// 2015-12-21 16:38:27.160 ReactiveCocoaLearn[83642:4836850] 接收信号=two
// 2015-12-21 16:38:27.162 ReactiveCocoaLearn[83642:4836850] 接收信号=one

相当于在发送某个信号之前先发送另一个信号

3-5、timeout
RACSignal *siganl = [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    
    // 假设某个请求的时间用了几秒
    [[RACScheduler  mainThreadScheduler] afterDelay:4 schedule:^{
    
        [subscriber sendNext:@"one"];
        [subscriber sendCompleted];
    }];

    return [RACDisposable disposableWithBlock:^{
//            NSLog(@"销毁信号");
    }];
    // 然后timeout就是当超过这个时间的时候就会出错
}] timeout:10.0 onScheduler:[RACScheduler mainThreadScheduler]];


[siganl subscribeNext:^(id x){

    
    NSLog(@"x==%@",x);

} error:^(NSError * error){

    // 这个地方就很容易来处理错误的时候啦
    NSLog(@"error==%@",[error description]);

} completed:^{

    NSLog(@"completed");
}];

比较适合用于 请求超时的时候

3-6、take & skip & takeLast
 RACSignal *siganl = [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {

    [subscriber sendNext:@"one"];
    [subscriber sendNext:@"two"];
    [subscriber sendNext:@"three"];
    [subscriber sendNext:@"four"];
    [subscriber sendCompleted];
    return [RACDisposable disposableWithBlock:^{

    }];
    
}] take:2];

[siganl subscribeNext:^(id x){

    NSLog(@"x==%@",x);

}];

//take 只接收前几次
//skip 跳过前几次
//takeLast 只接收最后几次
 / *
    takeUntilBlock:     
    takeWhileBlock:
    skipWhileBlock:
    skipUntilBlock:
  */

四、进阶使用

在我们向服务器进行请求的时候,RAC为我们带来了诸多方便的事情,值得探索。

此处还是用DeveloperLx的例子,textFiled举例说明。

4-1、throttle
[[self.testTextField.rac_textSignal throttle:0.5]subscribeNext:^(id x){
    NSLog(@"%@", x); 
}];

就是在我们设置那个时间内(0.5秒),不会发送消息,让其不会一直不断的发送过来。

4-2 distinctUntilChanged
   [[[self.testTextField.rac_textSignal throttle:0.5] distinctUntilChanged]subscribeNext:^(id x){

    NSLog(@"%@", x);
    
}];

相同的就不发送,直到有所该变再发送

4-3 ignore
  [[[[self.testTextField.rac_textSignal throttle:0.5] distinctUntilChanged] ignore:@""] subscribeNext:^(id x){

    NSLog(@"%@", x);
    
}];

忽略某个值,像上面就是忽略 空值

4-4 switchToLatest

先综合了下 map

 [[[[[[self.testTextField.rac_textSignal throttle:0.5] distinctUntilChanged] ignore:@""] map:^id(id value){

    return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber>subscriber){

        [subscriber sendNext:value];
        [subscriber sendCompleted];
    
        return [RACDisposable disposableWithBlock:^{}];
    }];

}]switchToLatest ]subscribeNext:^(NSString * x){

    NSLog(@"x==%@", x);
    
}];

只执行最后一次,这个地方有待推敲,暂时还不是很理解

4-5 merge
RACSignal * signalA = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

     
        [subscriber sendNext:@"Signal_A"];
        [subscriber sendCompleted];
    });
    
    return nil;
}];

RACSignal * signalB = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
     
   
        [subscriber sendNext:@"Signal_B"];
        [subscriber sendCompleted];
    });
    
    return nil;
}];

NSLog(@"开始预订");
[[RACSignal merge:@[signalA, signalB]]subscribeNext:^(id x) {
    
    NSLog(@"x==%@",x);
}];

 /*
  2015-12-21 17:54:24.105 ReactiveCocoaLearn[85576:4905054] 开始预订
  2015-12-21 17:54:26.306 ReactiveCocoaLearn[85576:4905054] x==Signal_A
  2015-12-21 17:54:27.398 ReactiveCocoaLearn[85576:4905054] x==Signal_B
  */

同时订阅信号

4-6 concat
NSLog(@"开始预订");
[[RACSignal concat:@[signalA, signalB]]subscribeNext:^(id x) {
    
    NSLog(@"x==%@",x);
}];

/*
  2015-12-21 17:57:03.718 ReactiveCocoaLearn[85651:4908056] 开始预订
  2015-12-21 17:57:05.720 ReactiveCocoaLearn[85651:4908056] x==Signal_A
  2015-12-21 17:57:09.012 ReactiveCocoaLearn[85651:4908056] x==Signal_B
*/

执行完A 后才执行 B ,而且A必须成功,B才会执行,他们是异步请求.

4-7、zipwith
NSLog(@"开始预订");
[[signalA zipWith:signalB] subscribeNext:^(id x) {
    
    NSLog(@"x==%@",x);
}];
  /*
  2015-12-21 18:01:18.770 ReactiveCocoaLearn[85742:4913279] 开始预订
  2015-12-21 18:01:22.071 ReactiveCocoaLearn[85742:4913279] x==<RACTuple: 0x7f8cc8c2c520> (
      "Signal_A",
      "Signal_B"
)
  */

注意看上面返回的时间差距
返回一个RACTuple(元祖) ,A、B 至少都发送过一次消息后,才返回。
三者以上的可以用下面这个,combineLatest,同上

[[RACSignal combineLatest:@[signalA,signalB,signalC]] subscribeNext:^(id x){

    NSLog(@"x==%@",x);
}];

五、RAC常见宏

5.1 RAC(TARGET, [KEYPATH, [NIL_VALUE]]):用于给某个对象的某个属性绑定

 RAC(self.testButton, backgroundColor) = [RACObserve(self.testButton, selected) map:^UIColor *(NSNumber * selected) {
    
    return [selected boolValue] ? [UIColor redColor] : [UIColor greenColor];
}];

[[self.testButton rac_signalForControlEvents:UIControlEventTouchUpInside]subscribeNext:^(UIButton * btn) {
    
    btn.selected = !btn.selected;
}];

直接改变button 的颜色

5.2 RACObserve(self, name):监听某个对象的某个属性,返回的是信号

[RACObserve(self.greenView, center) subscribeNext:^(id x) {
    
    NSLog(@"%@",x);
}];

点击按钮,改变其center之后

/*
  2015-12-21 18:18:52.229 ReactiveCocoaLearn[86031:4931305] NSPoint: {0, 0}
  2015-12-21 18:18:54.024 ReactiveCocoaLearn[86031:4931305] 按钮被点击了
  2015-12-21 18:18:54.025 ReactiveCocoaLearn[86031:4931305] NSPoint: {187.5, 333.5}
*/
下面这个也是同样的用这个宏的,这是用最少的代码写一个秒表。
RAC(self.testLabel, text) = [[RACSignal interval:1 onScheduler:[RACScheduler mainThreadScheduler]] map:^NSString *(NSDate * date) {
    
    return date.description;
}];

总的来说,记录的笔记大致差不多了,有很多东西自己还没深入了解,毕竟我还没运用在项目中,初次记录,慢慢学习吧。再次还是非常感谢DeveloperLx,让我了解RAC的这么好用的东东,后期继续探索中,暂时记录到此。

备注:

DeveloperLx 的github微博,在此。
另外参考了下列文章:
http://www.jianshu.com/p/87ef6720a096
http://southpeak.github.io/blog/2014/08/02/reactivecocoazhi-nan-%5B%3F%5D-:xin-hao/

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

推荐阅读更多精彩内容