ReactiveCocoa 框架学习——框架的使用

作者: Laughingg | 来源:发表于2015-10-01 12:31 被阅读2887次

    如何使用ReactiveCocoa

    ReactiveCocoa可以在iOS和OS X的应用开发中使用,对于iOS开发者,可以将RAC源码下载编译后,使用编译好的libReactiveCocoa-iOS.a文件。
    开发者也可以用CocoaPods来设置目标工程对ReactiveCocoa的依赖,只需要编辑Podfile文件,增加如下内容即可:

    pod 'ReactiveCocoa', '~> 4.0.2-alpha-1'
    
    Paste_Image.png

    一般会导入不成功
    ** 报错提示信息**


    Paste_Image.png

    需要在podfile加上use_frameworks,重新pod install 才能导入成功(这个是应为框架是swift的缘故)

    use_frameworks!
    
    Paste_Image.png

    ReactiveCocoa的特点

    RAC在应用中大量使用了block,由于Objective-C语言的内存管理是基于引用计数 的,为了避免循环引用问题,在block中如果要引用self,需要使用@weakify(self)@strongify(self)来避免强引用。另外,在使用时应该注意block的嵌套层数,不恰当的滥用多层嵌套block可能给程序的可维护性带来灾难。
    RAC的编程方式和传统的MVC方式差异巨大,所以需要较长的学习时间。

    对于一个常用框架的学习先要了解常用类和项目的基本结构

    ReactiveCocoa框架的结构 图(图来自刀哥分享, 刀哥Github

    psb.png

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

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

    先通过一个简单的示例来进行学习:


    Paste_Image.png

    将下面的代码添加到viewDidLoad方法的最后:

    // rac_textSignal 是一个通过分类实现的信号。subscribeNext 是订阅信号。
    // ReactiveCocoa框架使用category来为很多基本UIKit控件添加signal。这样你就能给控件添加订阅了,text field的rac_textSignal就是这么来的。
    [self.usernameTextField.rac_textSignal subscribeNext:^(id x){
      NSLog(@"%@", x);
    }];
    

    编译运行,在用户名输入框中输几个字。注意console的输出应该和下面的类似。


    Snip20151001_3.png
    2015-10-01 21:37:04.619 ReactiveCocoaDemo[4212:72362] 1
    2015-10-01 21:37:05.967 ReactiveCocoaDemo[4212:72362] 12
    2015-10-01 21:37:06.636 ReactiveCocoaDemo[4212:72362] 123
    2015-10-01 21:37:07.543 ReactiveCocoaDemo[4212:72362] 1234
    2015-10-01 21:37:08.620 ReactiveCocoaDemo[4212:72362] 12345
    2015-10-01 21:37:09.079 ReactiveCocoaDemo[4212:72362] 123456
    2015-10-01 21:37:09.488 ReactiveCocoaDemo[4212:72362] 1234567
    2015-10-01 21:37:10.120 ReactiveCocoaDemo[4212:72362] 12345678
    
    

    ReactiveCocoa signal(RACSignal)发送事件流给它的subscriber。目前总共有三种类型的事件:next、error、completed。一个signal在因error终止或者完成前,可以发送任意数量的next事件。在本教程的第一部分,我们将会关注next事件。

    RACSignal有很多方法可以来订阅不同的事件类型。每个方法都需要至少一个block,当事件发生时就会执行block中的逻辑。在上面的例子中可以看到每次next事件发生时,subscribeNext:方法提供的block都会执行。

    ReactiveCocoa有很多操作来控制事件流。假设你只关心超过3个字符长度的用户名,那么你可以使用filter操作来实现这个目的。把之前加在viewDidLoad中的代码更新成下面的:

    // filter :过滤器 将信号进行过滤,从而得到我们想要的信号。   subscribeNext :订阅信号。 
    **只有订阅了的信号,信号流才会起作用。**
    [[self.usernameTextField.rac_textSignal
        filter:^BOOL(id value){
             NSString*text = value;
             return text.length > 3;
        }]
            subscribeNext:^(id x){
            NSLog(@"%@", x);
      }];
    
    Paste_Image.png
    // 这个代码是不会起作用的。
    [self.usernameTextField.rac_textSignal
         filter:^BOOL(id value){
             NSString*text = value;
             return text.length > 3;
         }];
    

    响应式编程的本质是响应链条(也就是信号管道),根据响应链条来表达应用的功能。


    Paste_Image.png

    从上面的图中可以看到,rac_textSignal是起始事件。然后数据通过一个filter,如果这个事件包含一个长度超过3的字符串,那么该事件就可以通过。管道的最后一步就是subscribeNext:,block在这里打印出事件的值。

    filter操作的输出也是RACSignal,这点先放到一边。你可以像下面那样调整一下代码来展示每一步的操作。

    // 这个是将上面的信号管道进行拆分
    // 1. 先获取  self.usernameTextField.rac_textSignal 信号
    RACSignal *usernameSourceSignal =
        self.usernameTextField.rac_textSignal;
     
    // 2. 在过滤信号
    RACSignal *filteredUsername =[usernameSourceSignal
      filter:^BOOL(id value){
        NSString*text = value;
        return text.length > 3;
      }];
     
    // 3. 订阅信号
    [filteredUsername subscribeNext:^(id x){
      NSLog(@"%@", x);
    }];
    

    RACSignal的每个操作都会返回一个RACsignal,这在术语上叫做连贯接口(fluent interface)。这个功能可以让你直接构建管道,而不用每一步都使用本地变量。

    [[[self.usernameTextField.rac_textSignal
        // 转换信号 将textSignal 转换为NSMumber
        map:^id(NSString*text){
          return @(text.length);
      }]
        // 过滤信号
        filter:^BOOL(NSNumber*length){
          return[length integerValue] > 3;
      }]
         // 订阅信号
         subscribeNext:^(id x){
          NSLog(@"%@", x);
      }];
    
    Paste_Image.png

    新加的map操作通过block改变了事件的数据。map从上一个next事件接收数据,通过执行block把返回值传给下一个next事件。在上面的代码中,map以NSString为输入,取字符串的长度,返回一个NSNumber

    Paste_Image.png

    能看到map操作之后的步骤收到的都是NSNumber实例。你可以使用map操作来把接收的数据转换成想要的类型,只要它是个对象。

    创建一些信号,来表示用户名和密码输入框中的输入内容是否有效

        // 信号是否有效并返回信号
        RACSignal *validUsernameSignal =
        [self.usernameTextField.rac_textSignal
         map:^id(NSString *text) {
             return @([self isValidUsername:text]);
         }];
        RACSignal *validPasswordSignal =
        [self.passwordTextField.rac_textSignal
         map:^id(NSString *text) {
             return @([self isValidPassword:text]);
         }];
        
    
        // 将信号进行转换 并订阅信号
        [[validUsernameSignal  map:^id(NSNumber *usernameValid){
              return[usernameValid boolValue] ? [UIColor clearColor]:[UIColor yellowColor];
          }]  subscribeNext:^(UIColor *color){
              self.usernameTextField.backgroundColor = color;
         }];
        
        [[validPasswordSignal  map:^id(NSNumber *passwordValid){
              return[passwordValid boolValue] ? [UIColor clearColor]:[UIColor yellowColor];
          }]  subscribeNext:^(UIColor *color){
              self.passwordTextField.backgroundColor = color;
         }];
    

    上例代码的简化

    RAC(self.passwordTextField, backgroundColor) = [validPasswordSignal  map:^id(NSNumber *passwordValid){
          return[passwordValid boolValue] ? [UIColor clearColor]:[UIColor yellowColor];
        }];
     
    RAC(self.usernameTextField, backgroundColor) = [validUsernameSignal  map:^id(NSNumber *passwordValid){
         return[passwordValid boolValue] ? [UIColor clearColor]:[UIColor yellowColor];
       }];
    

    RAC宏允许直接把信号的输出应用到对象的属性上。RAC宏有两个参数,第一个是需要设置属性值的对象,第二个是属性名。每次信号产生一个next事件,传递过来的值都会应用到该属性上。

    Paste_Image.png
    • 注意:
    • 信号类(RACSiganl),只是表示当数据改变 时,信号内部会发出数据,它本身不具备发送信号的能力,而是交给内部一个订阅者去发出。
    • 默认一个信号都是冷信号,也就是值改变了,也不会触发,只有订阅了这个信号,这个信号才会变为热信号,值改变了才会触发。
    • 如何订阅信号:调用信号RACSignal的subscribeNext就能订阅。

    学习博客原文推荐:
    ReactiveCocoa - iOS开发的新框架 ——唐巧的技术博客
    最快让你上手ReactiveCocoa之基础篇——袁铮
    ReactiveCocoa入门教程——第一部分——BenBeng

    相关文章

      网友评论

        本文标题:ReactiveCocoa 框架学习——框架的使用

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