美文网首页
RAC自己练习下

RAC自己练习下

作者: 考槃在涧 | 来源:发表于2017-11-27 14:49 被阅读31次

    ReactiveCocoa使用个人总结

    ReactiveCocoa简介

    ReactiveCocoa(简称RAC)是由Github开源的一个应用于iOS和OS开发的新框架。

    ReactiveCocoa的作用

    在我们iOS开发过程中,当某些事件响应的时候,需要处理某些业务逻辑,这些事件都会用不同的方式来处理。比如:按钮的点击使用action、UIScrollView的滚动使用的delegate、监测属性的值改变使用KVO等系统提供的方式。

    RAC就是把这些不同的响应事件的方式统一起来,使这些方式都可以通过RAC来处理。

    RAC为事件提供了很多的处理方法,而且利用RAC来处理事件很方便,可以把要处理的事情和监听的代码放一起,方便我们的管理,这样就不需要跳转到对应的方法里去处理,非常符合我们开发中高内聚,低耦合的思想。

    RAC在某些特定情况下开发时可以大大简化代码,提高开发效率,并且是安全可靠的。

    配置RAC环境

    使用cocoapods来获取github上的开源库:

    pod 'ReactiveCocoa'
    

    如何使用RAC

    • RACSignal

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

    • RACSignal只是表示当数据改变时,信号内部会发出数据,他本身不具备发送信号的能力,而是交给内部的一个订阅者subscriber发出。
    • 默认一个信号都是冷信号,也就是说值改变了也不会触发,只有订阅了这个信号,值改变了才会触发。
    • 订阅信号:调用RACSignal的subscribeNext:方法就可以订阅信号。
        // 创建一个RACSignal信号
        RACSignal * signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
                      // 每当有订阅者订阅信号  就会调用这个block
                      // 发送信号
                      [subscriber sendNext:@"Hello"];
                      // 如果不再发送数据,最好调用此方法发送信号完成
                      [subscriber sendCompleted];
                      return [RACDisposable disposableWithBlock:^{
                            // 当信号发送完成或者发送错误,就会自动执行这个block,取消订阅信号
                            NSLog(@"信号被销毁");
                }];
        }];
        
        // 订阅信号,才会激活信号
        [signal subscribeNext:^(id x){
            // 每当有信号发出数据  就会调用block
            NSLog(@"接收到数据%@",x);
        }]
    
    
    • target-action

    RAC最基本的使用就是对事件的监听。我们来看以下代码:

    RACSignal * textFieldSignal = [textField rac_signalForControlEvents:UIControlEventEditingChanged];
    [textFieldSignal subscribeNext:^(id x) {
            NSLog(@"变化的text:%@",x);
    }];
    

    这段代码实现了监听UITextFieldUIControlEventEditingChanged事件,当textField里的内容发生变化时,发送变化的数据调用block种的内容将变化的内容实时打印出来。这两段代码通常会连在一起写:

    [[textField rac_signalForControlEvents:UIControlEventEditingChanged] subscribeNext:^(id x) {
            NSLog(@"变化的text:%@",x);
        }];
    

    同样的,我们以这段代码为模板进行RAC的使用,UIButton的点击事件我们也可以用RAC进行监测,添加方法,对于UITextField的文字更改的监听也有更简单的写法:

    // UIButton添加RAC监听点击方法
    [[button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
            NSLog(@"点击button:%@",x);
    }];
    // 对于UITextField 文字更改的监听
    [textField.rac_textSignal subscribeNext:^(id x) {
            NSLog(@"变化的text:%@",x);
    }]
    

    RAC的一些常见操作方法

    1. ReactiveCocoa操作方法之映射(flattenMap,Map)

      • flattenMap,Map用于把源信号内容映射成新的内容。

      flattenMap的使用:把源信号的内容映射成一个新的信号,信号可以是任意类型 flattenMap中的block返回的是信号

    ```
    // 创建信号中的信号
    

    RACSubject * signalOfSignals = [RACSubject subject];
    RACSubject * signals = [RACSubject subject];

    [[signalOfSignals flattenMap:^RACStream *(id value) {
        // 当signalOfsignals 发出signals 信号才会调用  这里的value返回的是信号
        return value;
    }] subscribeNext:^(id x) {
        // 只有signalOfSignals发出的signals也发出了内容(也就是最后一行代码执行)才会调用
        NSLog(@"=========>%@",x);
    }];
    
    // 信号中的信号发送一个信号
    [signalOfSignals sendNext:signals];
    // 信号发送一个内容
    [signals sendNext:@"信号发送信号"];
    
    ```
    
    > map的使用:把源信号的值映射成一个新的值,`map`中的block返回的是一般对象(这里一般用于把源信号里的json数据映射成相应的model)
    
    ```
    // 监听文本框内容的改变,映射成一个新的值
    [[textField.rac_textSignal map:^id(id value) {
        // 这里是把文本框的内容加上 “Hello RAC” 的前缀
        return [NSString stringWithFormat:@"Hello RAC :%@",value];
    }] subscribeNext:^(id x) {
        NSLog(@"%@",x);
    }];
    ```
    
    > 开发中,如果信号发出的不是信号,映射一般使用`map`,如果信号发出的是信号,则使用`flattenMap`。
    
    1. ReactiveCocoa操作方法之组合

      • concat:按一定顺序拼接信号,当多个信号发出的时候,有顺序的接收信号
       RACSignal * signalA = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
          [subscriber sendNext:@"First"];
          [subscriber sendCompleted];
          return nil;
      }];
      
      RACSignal * signalB = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
          [subscriber sendNext:@"Second"];
          [subscriber sendCompleted];
          return nil;
      }];
      
      // 把A拼接到B后   A发送完成B才会被激活  按顺序
      RACSignal * concatSignal = [signalA concat:signalB];
      [concatSignal subscribeNext:^(id x) {
          NSLog(@"Concat Signal %@",x);
      }];
      
      • then:用于连接两个信号,当第一个信号完成,才会连接then返回的信号,第一个信号会被忽略掉。
       RACSignal * thenSignal = [signalA then:^RACSignal *{
          return signalB;
      }];
      
      [thenSignal subscribeNext:^(id x) {
          // 此处输出为 Then signal Second   第一个信号会被忽略掉
          NSLog(@"Then signal %@",x);
      }];
      
      • merge把多个信号合并为一个信号,任何一个信号有新值就会调用。
      RACSignal * mergeSignal = [signalA merge:signalB];
      [mergeSignal subscribeNext:^(id x) {
          NSLog(@"Merge Signal %@",x);
      }];
      
      • zipWith:用于把两个信号压缩成一个信号,只有两个信号都发出信号内容时,把两个信号的内容合并成一个元组,才会触发压缩流的next事件。
      RACSignal * zipWithSignal = [signalA zipWith:signalB];
      [zipWithSignal subscribeNext:^(id x) {
          NSLog(@"ZipWith Signal %@",x);
      }];
      
      • combineLatest:将多个信号合并起来,并且拿到各个信号的最新的值,必须每个合并的signal至少都有过一次sendNext,才会触发合并的信号。这个方法试了一下,觉得跟zipWith是相似的。
      RACSignal * combineSignal = [signalA combineLatestWith:signalB];
      [combineSignal subscribeNext:^(id x) {
          NSLog(@"Combine Signal %@",x);
      }];
      
      • reduce:聚合:用于信号发出的内容是元组,把信号发出元组的值聚合成一个值
      // 常见的方法(先组合再聚合)。
      // 先组合再聚合,把上面两个信号聚合在一起。形成一个信号  (两者都成立才为yes)
      // 用于登录页面,两个文本框的内容都符合条件,登录按钮才会响应
          RACSignal * reduceSignal = [RACSignal combineLatest:@[usernameLengthSignal, passwordLengthSignal] reduce:^id(NSNumber * username , NSNumber * password){
              return @(username.boolValue && password.boolValue);
          }];
          
      

    RAC一些常用的宏

    1. RAC(),这个宏的完整写法是:RAC(TARGET,[KEYPATH,NIL_VALUE])

      使用方法:

       RAC(label,text) = self.textField.rac_textSignal;
       RAC(label,text,@"如果是nil就显示这里的文字") = self.textField.rac_textSignal;
      

      最常用的宏RAC()一般出现在等号左边,右边是一个RACSignal,作用是将一个对象(这里是label)的一个属性(这里是label的text)和一个RACSignal绑定,RACSignal每产生一个value,都会自动将这个对象的这个属性设置为value。如果value是nil,就设置为第三个参数。(上面的代码的效果是把label的text和textField的文字变动的signal绑定, 每当textField里面的文字有变化,就给label设置新的text。)

    2. RACObserve(TARGET,KEYPATH)

      作用是观察TARGET的KEYPATH属性,产生一个RACSignal,相当于KVO。

      RAC(self.outputLabel, text) = RACObserve(self.model, name); 
      

      把label的text和model的name绑定,只要name有变化,label的text也会变化为相应的值。

    暂时写这么点吧。 代码点击这里.

    相关文章

      网友评论

          本文标题:RAC自己练习下

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