ReactiveCocoa 带你装B,带你飞

作者: 我是七月 | 来源:发表于2016-12-29 15:36 被阅读300次

    函数式、响应式编程(Functional Reactive Programming)现在很火。

    ReactiveCocoa

    什么是RAC?

    几乎每一篇介绍RAC的文章开头都是这么一个问题。我这篇文章是写给新手(包括我自己)看的,所以这个问题更是无法忽视。
    简单的说,RAC就是一个第三方库,他可以大大简化你的代码过程。
    官方的说,ReactiveCocoa(其简称为RAC)是由GitHub开源的一个应用于iOS和OS X开发的新框架。RAC具有函数式编程响应式编程的特性。

    为什么我们要学习RAC?

    为了提高我们的开发效率。RAC在某些特定情况下开发时可以大大简化代码,并且目前来看安全可靠。

    配置RAC环境

    我习惯用cocoapods来安装github上得开源库

    platform:ios, '8.0'
    pod 'ReactiveCocoa','~>2.1.8'
    

    这里有一点要注意下就是RAC的版本问题,最新版的RAC已经支持Swift了,但是在OC的程序安装最新版的RAC可能跑不起来,所以推荐大家使用2.5.0版本以下的RAC(具体支持Swift的版本可能有误,但我引用的2.1.8肯定是没问题的)。

    注意:podfile如果只描述pod 'ReactiveCocoa','~>2.1.8',会导入不成功。

    • 报错提示信息


      报错提示信息
    • 需要在pod file加上use_frameworks!,重新pod install 才能导入成功。


      注意:加上use_frameworks

    最后导入头文件
    建议在pch导入,全局使用。

    #import <ReactiveCocoa/ReactiveCocoa.h>
    
    

    RAC使用

    UIButton单击事件
    [[self.button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
        NSLog(@"按钮单击");
    }];
    
    监视UITextField内容变化
       [[self.field rac_textSignal] subscribeNext:^(id x) {
        NSLog(@"UITextField: %@", x); 
       }];
    
       [[fid.rac_textSignal map:^id(id value) {
            DDLog(@"value===%@", value);
            return @1;
        }] subscribeNext:^(id x) {
            DDLog(@"x===%@", x);
        }];
        
        
        [[fid.rac_textSignal filter:^BOOL(NSString *value) {
            return [value length] > 3;
        }] subscribeNext:^(id x) {
            NSLog(@"filterlllll= %@", x);
        }];
    
    监视UISwitch值变化
    [[self.switchControl rac_newOnChannel] subscribeNext:^(id x) {
        NSLog(@"UISwitch: %@", x);
    }];
    
    UIAlertView按钮监听
       UIAlertView *alert = [[UIAlertView alloc]initWithTitle:@"提示" message:@"测测" delegate:self cancelButtonTitle:@"取消" otherButtonTitles:@"确定", nil];
        [alert show];
    
        [alert.rac_buttonClickedSignal subscribeNext:^(id x) {
            DDLog(@"alert====%@",x);
        }];
        
    
    通知
    [[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationDidBecomeActiveNotification object:nil] subscribeNext:^(id x) {
        NSLog(@"UIApplicationDidBecomeActiveNotification");
    }];
    
    定时器
    • 延时执行
     //五秒后执行一次
      [[RACScheduler mainThreadScheduler]afterDelay:5 schedule:^{
          NSLog(@"五秒后执行一次");
      }];
    
    • 定时执行
      //每隔两秒执行一次
      //这里要加takeUntil条件限制一下否则当控制器pop后依旧会执行
      [[[RACSignal interval:2 onScheduler:[RACScheduler mainThreadScheduler]] takeUntil:self.rac_willDeallocSignal ] subscribeNext:^(id x) {
          NSLog(@"每两秒执行一次");
      }];
    
    对其他对象的keyPath的值跟踪--KVO
    • UIView.frame
    [RACObserve(self.view, frame) subscribeNext:^(id x) {
        NSLog(@"self.view.frame: %@", x);
    }];
    
    • UIScrollView.contentOffset
    [RACObserve(scrollView, contentOffset) subscribeNext:^(id x) {
        NSLog(@"scrollView.contentOffset: %@", x);
    }];
    
    ReactiveCocoa常见宏。
    • RAC(TARGET, [KEYPATH, [NIL_VALUE]]):用于给某个对象的某个属性绑定。可以看作某个属性的值与一些信号的联动
      // 只要文本框文字改变,就会修改label的文字
        RAC(self.labelView,text) = _textField.rac_textSignal;
    
    • RACObserve(self, name):监听某个对象的某个属性,返回的是信号。监听属性的改变,使用block的KVO
     [RACObserve(lbl, text) subscribeNext:^(id x) {
            NSLog(@"RACObserve===%@",x);
        }];
    

    对上面ReactiveCocoa开发中常见用法总结如下:

    1. 代替代理.

    rac_signalForSelector:用于替代代理。

    2. 代替KVO :

    rac_valuesAndChangesForKeyPath:用于监听某个对象的属性改变。

    3. 监听事件:

    rac_signalForControlEvents:用于监听某个事件。

    4.代替通知:

    rac_addObserverForName:用于监听某个通知。

    5.监听文本框文字改变:

    rac_textSignal:只要文本框发出改变就会发出这个信号。

    6. 处理当界面有多次请求时,需要都获取到数据时,才能展示界面

    rac_liftSelector:withSignalsFromArray:Signals:当传入的Signals(信号数组),每一个signal都至少sendNext过一次,就会去触发第一个selector参数的方法。
    使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。

    示例代码

     // 1.代替代理
        // 需求:自定义redView,监听红色view中按钮点击
        // 之前都是需要通过代理监听,给红色View添加一个代理属性,点击按钮的时候,通知代理做事情
        // rac_signalForSelector:把调用某个对象的方法的信息转换成信号,就要调用这个方法,就会发送信号。
        // 这里表示只要redV调用btnClick:,就会发出信号,订阅就好了。
        [[redV rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(id x) {
            NSLog(@"点击红色按钮");
        }];
    
      // 2.KVO
        // 把监听redV的center属性改变转换成信号,只要值改变就会发送信号
        // observer:可以传入nil
        [[redV rac_valuesAndChangesForKeyPath:@"center" options:NSKeyValueObservingOptionNew observer:nil] subscribeNext:^(id x) {
            NSLog(@"%@",x);
        }];
    
      // 3.监听事件
        // 把按钮点击事件转换为信号,点击按钮,就会发送信号
        [[self.btn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
           NSLog(@"按钮被点击了");
        }];
    
      // 4.代替通知
        // 把监听到的通知转换信号
        [[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(id x) {
            NSLog(@"键盘弹出");
        }];
    
      // 5.监听文本框的文字改变
       [_textField.rac_textSignal subscribeNext:^(id x) {
           NSLog(@"文字改变了%@",x);
       }];
    
      // 6.处理多个请求,都返回结果的时候,统一做处理.
        RACSignal *request1 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
            // 发送请求1
            [subscriber sendNext:@"发送请求1"];
            return nil;
        }];
    
        RACSignal *request2 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
            // 发送请求2
            [subscriber sendNext:@"发送请求2"];
            return nil;
        }];
    
        // 使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
        [self rac_liftSelector:@selector(updateUIWithR1:r2:) withSignalsFromArray:@[request1,request2]];
    }
    // 更新UI(该方法有要求,有多少个信号就要求有多少个参数,参数的内容就是发送的数据。)
    - (void)updateUIWithR1:(id)data r2:(id)data1
    {
        NSLog(@"更新UI%@,%@",data,data1);
    }
    

    相关文章

      网友评论

        本文标题:ReactiveCocoa 带你装B,带你飞

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