如何使用ReactiveCocoa
ReactiveCocoa可以在iOS和OS X的应用开发中使用,对于iOS开发者,可以将RAC源码下载编译后,使用编译好的libReactiveCocoa-iOS.a文件。
开发者也可以用CocoaPods来设置目标工程对ReactiveCocoa的依赖,只需要编辑Podfile文件,增加如下内容即可:
pod 'ReactiveCocoa', '~> 4.0.2-alpha-1'
一般会导入不成功
** 报错提示信息**
需要在podfile加上use_frameworks,重新pod install 才能导入成功(这个是应为框架是swift的缘故)
use_frameworks!
ReactiveCocoa的特点
RAC在应用中大量使用了block,由于Objective-C语言的内存管理是基于引用计数 的,为了避免循环引用问题,在block中如果要引用self,需要使用@weakify(self)
和@strongify(self)
来避免强引用。另外,在使用时应该注意block的嵌套层数,不恰当的滥用多层嵌套block可能给程序的可维护性带来灾难。
RAC的编程方式和传统的MVC方式差异巨大,所以需要较长的学习时间。
对于一个常用框架的学习先要了解常用类和项目的基本结构
ReactiveCocoa框架的结构 图(图来自刀哥分享, 刀哥Github)
** RAC中最核心的类RACSiganl**
搞定RACSiganl,这个类就能用ReactiveCocoa开发了。
RACSiganl:信号类,一般表示将来有数据传递,只要有数据改变,信号内部接收到数据,就会马上发出数据。
先通过一个简单的示例来进行学习:
将下面的代码添加到viewDidLoad方法的最后:
// rac_textSignal 是一个通过分类实现的信号。subscribeNext 是订阅信号。
// ReactiveCocoa框架使用category来为很多基本UIKit控件添加signal。这样你就能给控件添加订阅了,text field的rac_textSignal就是这么来的。
[self.usernameTextField.rac_textSignal subscribeNext:^(id x){
NSLog(@"%@", x);
}];
编译运行,在用户名输入框中输几个字。注意console的输出应该和下面的类似。
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);
}];
// 这个代码是不会起作用的。
[self.usernameTextField.rac_textSignal
filter:^BOOL(id value){
NSString*text = value;
return text.length > 3;
}];
响应式编程的本质是响应链条(也就是信号管道),根据响应链条来表达应用的功能。
从上面的图中可以看到,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);
}];
新加的map操作通过block改变了事件的数据。map从上一个next事件接收数据,通过执行block把返回值传给下一个next事件。在上面的代码中,map以NSString为输入,取字符串的长度,返回一个NSNumber。
能看到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事件,传递过来的值都会应用到该属性上。
- 注意:
- 信号类(RACSiganl),只是表示当数据改变 时,信号内部会发出数据,它本身不具备发送信号的能力,而是交给内部一个订阅者去发出。
- 默认一个信号都是冷信号,也就是值改变了,也不会触发,只有订阅了这个信号,这个信号才会变为热信号,值改变了才会触发。
- 如何订阅信号:调用信号RACSignal的subscribeNext就能订阅。
学习博客原文推荐:
ReactiveCocoa - iOS开发的新框架 ——唐巧的技术博客
最快让你上手ReactiveCocoa之基础篇——袁铮
ReactiveCocoa入门教程——第一部分——BenBeng