iOS开发

ReactiveCocoa 入坑 ing

2015-04-28  本文已影响1760人  stillwalking

"A good litmus test for MVVM is whether you’re able to write automated tests for your UI behavior without actually having a live UI." --Justin Spahr-Summers.

FRP的核心是信号,信号在ReactiveCocoa中通过RACSignal来表示,信号是数据流,可以被绑定和传递。

用响应式编程的一个关键区别,你不需要使用实例变量来追踪瞬时状态。

ReactiveCocoa的核心就是信号,而它不过就是事件流。还能再更简单点吗?

可以把信号想象成水龙头,只不过里面不是水,而是玻璃球(value),直径跟水管的内径一样,这样就能保证玻璃球是依次排列,不会出现并排的情况(数据都是线性处理的,不会出现并发情况)。水龙头的开关默认是关的,除非有了接收方(subscriber),才会打开。这样只要有新的玻璃球进来,就会自动传送给接收方。可以在水龙头上加一个过滤嘴(filter),不符合的不让通过,也可以加一个改动装置,把球改变成符合自己的需求(map)。也可以把多个水龙头合并成一个新的水龙头(combineLatest:reduce:),这样只要其中的一个水龙头有玻璃球出来,这个新合并的水龙头就会得到这个球。

需要说明的是 因为RAC很大程度上是依赖于Block的.所以在RAC前面我们加上@weakify(my_variable) 避免循环引用,然后在每一个
RAC块中为了防止提前释放我们需要用
@strongify(my_variable)来对对象进行持有

最简单的:

[RACObserve(self,username) subscribeNext:^(NSString *newName){
    NSLog(@"Name changed to %@",newName);
}];  

RACObserve使用了KVO来监听property(username)的变化,只要username被自己或外部改变,block就会被执行
但不是所有的property都可以被RACObserve,该property必须支持KVO,比如NSURLCache的currentDiskUsage就不能被RACObserve。

Signal and Subscriber

Signal获取到数据后,会调用Subscriber的sendNext, sendComplete, sendError方法来传送数据给Subscriber,Subscriber自然也有方法来获取传过来的数据,如:[signal subscribeNext:error:completed]。这样只要没有sendComplete和sendError,新的值就会通过sendNext源源不断地传送过来.

ReactiveCocoa signal(RACSignal)发送事件流给它的subscriber。目前共有三种类型的事件:nexterrorcompleted。一个signal在因error终止或者完成前,可以发送任意数量的next事件.
RACSignal有很多方法可以来订阅不同的事件类型。每个方法都需要至少一个block,当事件发生时就会执行block中的逻辑。
ReactiveCocoa 框架使用 category 来为很多基本UIKit控件添加signal。这样就能给控件添加订阅了,比如 text field 的 rac_textSignal
RACSignal的每个操作都会返回一个RACsignal,这在术语上叫做连贯接口(fluent interface)。这个功能可以让你直接构建管道,而不用每一步都使用本地变量。

以下几段代码来自apem的翻译博文

  1. 只要 text 有变化, 将 text 发送给 Subscriber

    [self.usernameTextField.rac_textSignal subscribeNext:^(id x){
    NSLog(@"%@", x);
    }];

  2. 筛选长度 >3, 满足则发送给 Subscriber

     [[self.usernameTextField.rac_textSignal
      filter:^BOOL(NSString *text){
          return text.length > 3;
      }]
     subscribeNext:^(id x){
         NSLog(@"%@", x);
     }];
    
  3. 上一段代码的分步:

     RACSignal *usernameSourceSignal =
         self.usernameTextField.rac_textSignal;
       
     RACSignal *filteredUsername =[usernameSourceSignal
       filter:^BOOL(id value){
         NSString*text = value;
         return text.length > 3;
       }];
       
     [filteredUsername subscribeNext:^(id x){
       NSLog(@"%@", x);
     }];  
    
  4. 在管道中添加 map : 这里的 map 操作之后的步骤收到的都是NSNumber实例。可以使用map操作来把接收的数据转换成想要的类型,只要它是对象

     [[[self.usernameTextField.rac_textSignal
            map:^id(NSString*text){
                return @(text.length);
            }]
           filter:^BOOL(NSNumber*length){
               return[length integerValue] > 3;
           }]
          subscribeNext:^(id x){
              NSLog(@"%@", x);
          }];
    

text.length 返回 NSUInteger 基本类型。为了将它作为事件的内容,NSUInteger 必须被封装成对象。简单的封装方法 ——> @(text.length)

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

替换举例

简明语法入门,图文并茂:Functional Reactive Programming on iOS with ReactiveCocoa -- by Ash Furrow

Getting Started with ReactiveCocoa -- by Ash Furrow

Model-View-ViewModel for iOS--by Ash Furrow 译文点此

ReactiveCocoa与Functional Reactive Programming -- by Limboy

说说ReactiveCocoa 2 -- by Limboy

Reactive Cocoa Tutorial --by sunnyxx

ReactiveCocoa 用 RACSignal 替代 Delegate

工程实战:ReactiveCocoa入门教程——第一部分 -- by BenBeng 英文原文 -- by RayWenderlich

语法学习:ReactiveCocoa-Documentation-BasicOperators

ReactiveCocoa-Documentation-FrameworkOverview

使用ReactiveCocoa实现iOS平台响应式编程 这篇在网页的后半部分

上一篇下一篇

猜你喜欢

热点阅读