"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。目前共有三种类型的事件:next、error、completed。一个signal在因error终止或者完成前,可以发送任意数量的next事件.
RACSignal有很多方法可以来订阅不同的事件类型。每个方法都需要至少一个block,当事件发生时就会执行block中的逻辑。
ReactiveCocoa 框架使用 category 来为很多基本UIKit控件添加signal。这样就能给控件添加订阅了,比如 text field 的 rac_textSignal
RACSignal的每个操作都会返回一个RACsignal,这在术语上叫做连贯接口(fluent interface)。这个功能可以让你直接构建管道,而不用每一步都使用本地变量。
-
只要 text 有变化, 将 text 发送给 Subscriber
[self.usernameTextField.rac_textSignal subscribeNext:^(id x){
NSLog(@"%@", x);
}]; -
筛选长度 >3, 满足则发送给 Subscriber
[[self.usernameTextField.rac_textSignal filter:^BOOL(NSString *text){ return text.length > 3; }] subscribeNext:^(id x){ NSLog(@"%@", x); }];
-
上一段代码的分步:
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); }];
-
在管道中添加 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。
替换举例
-
代理delegate
rac_signalForControlEvents [[[[self.signInButton rac_signalForControlEvents:UIControlEventTouchUpInside] doNext:^(id x){ //doNext,side effect,block并没有返回值。�是附加操作,并不改变事件本身 self.signInButton.enabled =NO; self.signInFailureText.hidden =YES; }] flattenMap:^id(id x){ return [self signInSignal]; }] subscribeNext:^(NSNumber *signedIn){ BOOL success =[signedIn boolValue]; self.signInFailureText.hidden = success; if(success){ [self performSegueWithIdentifier:@"signInSuccess" sender:self]; } }];
事件event
。。。
etc
简明语法入门,图文并茂: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平台响应式编程 这篇在网页的后半部分