函数式、响应式编程(Functional Reactive Programming)现在很火。
什么是RAC?
几乎每一篇介绍RAC的文章开头都是这么一个问题。我这篇文章是写给新手(包括我自己)看的,所以这个问题更是无法忽视。
简单的说,RAC就是一个第三方库,他可以大大简化你的代码过程。
官方的说,ReactiveCocoa(其简称为RAC)是由GitHub开源的一个应用于iOS和OS X开发的新框架。RAC具有函数式编程和响应式编程的特性。
为什么我们要学习RAC?
为了提高我们的开发效率。RAC在某些特定情况下开发时可以大大简化代码,并且目前来看安全可靠。
配置RAC环境
我习惯用cocoapods来安装github上得开源库
platform:ios, '8.0'
pod 'ReactiveCocoa','~>2.1.8'
这里有一点要注意下就是RAC的版本问题,最新版的RAC已经支持Swift了,但是在OC的程序安装最新版的RAC可能跑不起来,所以推荐大家使用2.5.0版本以下的RAC(具体支持Swift的版本可能有误,但我引用的2.1.8肯定是没问题的)。
注意:podfile如果只描述pod 'ReactiveCocoa','~>2.1.8',会导入不成功。
-
报错提示信息
-
需要在pod file加上use_frameworks!,重新pod install 才能导入成功。
最后导入头文件
建议在pch导入,全局使用。
#import <ReactiveCocoa/ReactiveCocoa.h>
RAC使用
UIButton单击事件
[[self.button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
NSLog(@"按钮单击");
}];
监视UITextField内容变化
[[self.field rac_textSignal] subscribeNext:^(id x) {
NSLog(@"UITextField: %@", x);
}];
[[fid.rac_textSignal map:^id(id value) {
DDLog(@"value===%@", value);
return @1;
}] subscribeNext:^(id x) {
DDLog(@"x===%@", x);
}];
[[fid.rac_textSignal filter:^BOOL(NSString *value) {
return [value length] > 3;
}] subscribeNext:^(id x) {
NSLog(@"filterlllll= %@", x);
}];
监视UISwitch值变化
[[self.switchControl rac_newOnChannel] subscribeNext:^(id x) {
NSLog(@"UISwitch: %@", x);
}];
UIAlertView按钮监听
UIAlertView *alert = [[UIAlertView alloc]initWithTitle:@"提示" message:@"测测" delegate:self cancelButtonTitle:@"取消" otherButtonTitles:@"确定", nil];
[alert show];
[alert.rac_buttonClickedSignal subscribeNext:^(id x) {
DDLog(@"alert====%@",x);
}];
通知
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationDidBecomeActiveNotification object:nil] subscribeNext:^(id x) {
NSLog(@"UIApplicationDidBecomeActiveNotification");
}];
定时器
- 延时执行
//五秒后执行一次
[[RACScheduler mainThreadScheduler]afterDelay:5 schedule:^{
NSLog(@"五秒后执行一次");
}];
- 定时执行
//每隔两秒执行一次
//这里要加takeUntil条件限制一下否则当控制器pop后依旧会执行
[[[RACSignal interval:2 onScheduler:[RACScheduler mainThreadScheduler]] takeUntil:self.rac_willDeallocSignal ] subscribeNext:^(id x) {
NSLog(@"每两秒执行一次");
}];
对其他对象的keyPath的值跟踪--KVO
- UIView.frame
[RACObserve(self.view, frame) subscribeNext:^(id x) {
NSLog(@"self.view.frame: %@", x);
}];
- UIScrollView.contentOffset
[RACObserve(scrollView, contentOffset) subscribeNext:^(id x) {
NSLog(@"scrollView.contentOffset: %@", x);
}];
ReactiveCocoa常见宏。
-
RAC(TARGET, [KEYPATH, [NIL_VALUE]])
:用于给某个对象的某个属性绑定。可以看作某个属性的值与一些信号的联动
// 只要文本框文字改变,就会修改label的文字
RAC(self.labelView,text) = _textField.rac_textSignal;
-
RACObserve(self, name)
:监听某个对象的某个属性,返回的是信号。监听属性的改变,使用block的KVO
[RACObserve(lbl, text) subscribeNext:^(id x) {
NSLog(@"RACObserve===%@",x);
}];
对上面ReactiveCocoa开发中常见用法总结如下:
1. 代替代理.
rac_signalForSelector:用于替代代理。
2. 代替KVO :
rac_valuesAndChangesForKeyPath:用于监听某个对象的属性改变。
3. 监听事件:
rac_signalForControlEvents:用于监听某个事件。
4.代替通知:
rac_addObserverForName:用于监听某个通知。
5.监听文本框文字改变:
rac_textSignal:只要文本框发出改变就会发出这个信号。
6. 处理当界面有多次请求时,需要都获取到数据时,才能展示界面
rac_liftSelector:withSignalsFromArray:Signals:当传入的Signals(信号数组),每一个signal都至少sendNext过一次,就会去触发第一个selector参数的方法。
使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
示例代码
// 1.代替代理
// 需求:自定义redView,监听红色view中按钮点击
// 之前都是需要通过代理监听,给红色View添加一个代理属性,点击按钮的时候,通知代理做事情
// rac_signalForSelector:把调用某个对象的方法的信息转换成信号,就要调用这个方法,就会发送信号。
// 这里表示只要redV调用btnClick:,就会发出信号,订阅就好了。
[[redV rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(id x) {
NSLog(@"点击红色按钮");
}];
// 2.KVO
// 把监听redV的center属性改变转换成信号,只要值改变就会发送信号
// observer:可以传入nil
[[redV rac_valuesAndChangesForKeyPath:@"center" options:NSKeyValueObservingOptionNew observer:nil] subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
// 3.监听事件
// 把按钮点击事件转换为信号,点击按钮,就会发送信号
[[self.btn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
NSLog(@"按钮被点击了");
}];
// 4.代替通知
// 把监听到的通知转换信号
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(id x) {
NSLog(@"键盘弹出");
}];
// 5.监听文本框的文字改变
[_textField.rac_textSignal subscribeNext:^(id x) {
NSLog(@"文字改变了%@",x);
}];
// 6.处理多个请求,都返回结果的时候,统一做处理.
RACSignal *request1 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
// 发送请求1
[subscriber sendNext:@"发送请求1"];
return nil;
}];
RACSignal *request2 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
// 发送请求2
[subscriber sendNext:@"发送请求2"];
return nil;
}];
// 使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
[self rac_liftSelector:@selector(updateUIWithR1:r2:) withSignalsFromArray:@[request1,request2]];
}
// 更新UI(该方法有要求,有多少个信号就要求有多少个参数,参数的内容就是发送的数据。)
- (void)updateUIWithR1:(id)data r2:(id)data1
{
NSLog(@"更新UI%@,%@",data,data1);
}