iOS DeveloperiOS学习笔记iOS开发记录

ReactiveCocoa 带你装B,带你飞

2016-12-29  本文已影响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',会导入不成功。

最后导入头文件
建议在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
[RACObserve(self.view, frame) subscribeNext:^(id x) {
    NSLog(@"self.view.frame: %@", x);
}];
[RACObserve(scrollView, contentOffset) subscribeNext:^(id x) {
    NSLog(@"scrollView.contentOffset: %@", x);
}];
ReactiveCocoa常见宏。
  // 只要文本框文字改变,就会修改label的文字
    RAC(self.labelView,text) = _textField.rac_textSignal;
 [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);
}
上一篇下一篇

猜你喜欢

热点阅读