什么是RAC?
几乎每一篇介绍RAC的文章开头都是这么一个问题。我这篇文章是写给新手(包括我自己)看的,所以这个问题更是无法忽视。
简单的说,RAC就是一个第三方库,他可以大大简化你的代码过程。
官方的说,(其简称为RAC)是由GitHub开源的一个应用于iOS和OS X开发的新框架。RAC具有和的特性。
为什么我们要学习RAC?
为了提高我们的开发效率。RAC在某些特定情况下开发时可以大大简化代码,并且目前来看安全可靠。
配置RAC环境
我习惯用cocoapods来安装github上得开源库,不会的新手iOS开发者有兴趣可以去学一下。
想学习cocoapods的同学推荐。
1 2 | platform:ios, '8.0' pod 'ReactiveCocoa' , '~>2.1.8' |
这里有一点要注意下就是RAC的版本问题,由于还没学习Swift,所以我是用OC编写程序的,最新版的RAC已经支持Swift了,但是在OC的程序安装最新版的RAC可能跑不起来,所以推荐大家使用2.5.0版本以下的RAC(具体支持Swift的版本可能有误,但我引用的2.1.8肯定是没问题的)。
使用RAC
1.target-action
RAC最基本的入门使用技巧就是对事件的监听。
PS:在iOS开发中,我们所说的点击事件其实就是target-action,接触过iOS开发的人都不会陌生UIControlEventTouchUpInside,这就是按下并松开的动作。不仅仅是UIButton,还有UITextField也有目标-动作模式。
使用前别忘了引用头文件~
#import
接下来就是最关键的RAC代码了。
[[self.textFild rac_signalForControlEvents:UIControlEventEditingChanged] subscribeNext:^(id x){ NSLog(@"change"); }];
就这么短短的两行代码。他实现了一个功能,即监听了textFild的UIControlEventEditingChanged事件,当事件发生时实现方法NSLog
。
add Target
了。对于textFild的文字更改监听也有更简单的写法 [[self.textFild rac_textSignal] subscribeNext:^(id x) { NSLog(@"%@",x); }];
这样就是每次改变TextFild都输出改变后的结果。
再比如给我们的某个label添加一个手势动作,我们也可以用简单的RAC代码完成
1 2 3 4 5 | UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc] init]; [[tap rac_gestureSignal] subscribeNext:^( id x) { NSLog (@ "tap" ); }]; [ self .view addGestureRecognizer:tap]; |
这段具体我就不解释了,相信大家都能看得懂,看不懂的自己写写就懂了。
2.代理
用RAC写代理是有局限的,它只能实现返回值为void的代理方法
首先我们要明白我们为什么要用RAC写代理?答:简化代码!是的,的确为了简化代码,为什么我要再这里强调这个,是因为在代理方法中我深深的感受到了RAC的优点。一开始我也不愿意花功夫去学RAC,但是我师父给我举了一个例子,如果一个View里有多个AlertView,每个AlertView有很多个按钮,每个按钮都有自己的点击事件,我应该怎么写?我想了一下,不但每个按钮需要打标记,而且每个AlertView也要打标记,然后再往代理点击事件里加各种方法,代码就又臭又长。那么让我们看看RAC怎么写代理方法。
1 2 3 4 5 6 7 | UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@ "RAC" message:@ "RAC TEST" delegate: self cancelButtonTitle:@ "cancel" otherButtonTitles:@ "other" , nil ]; [[ self rac_signalForSelector: @selector (alertView:clickedButtonAtIndex:) fromProtocol: @protocol (UIAlertViewDelegate)] subscribeNext:^(RACTuple *tuple) { NSLog (@ "%@" ,tuple.first); NSLog (@ "%@" ,tuple.second); NSLog (@ "%@" ,tuple.third); }]; [alertView show]; |
我们来看RAC的语句。@selector
是指这次事件监听的方法fromProtocol
指依赖的代理。这里block中有一个RACTuple,他相当于是一个集合类,他下面的first,second等就是类的各个参数,我这里点了AlertView第二个按钮other输出了一下。
1 2 3 | 2016-01-04 18:24:29.114 RACStudyTest[5003:388870] <UIAlertView: 0x7ff260c90c70; frame = (0 0; 0 0); layer = <CALayer: 0x7ff260c91030>> 2016-01-04 18:24:29.115 RACStudyTest[5003:388870] 1 2016-01-04 18:24:29.115 RACStudyTest[5003:388870] (null) |
可以看出tuple.second
是ButtonAtIndex中Button的序号。那么对于上面那个我举的例子,就可以用switch
给各个按钮添加方法,这样的代码看起来更容易理解,方面后期维护。
当然了,AlertView代理也有简化的代码。
1 2 3 | [[alertView rac_buttonClickedSignal] subscribeNext:^( id x) { NSLog (@ "%@" ,x); }]; |
这里的x就是各个Button的序号了,可以直接应对我上述遇到的问题。
3.通知
在我们的开发中通知也是一个比较常用的功能,主要的应用场景是某个页面进行数据重传需要更新model但是点击返回栈时不会刷新返回界面的数据,这时就可以用通知来更新另一个页面的数据,当然我们也可以在另一个页面的ViewDidAppear
方法中刷新数据,但那是题外话。
这里写的Demo就是我上述说的情况。
首先,在某个页面中我们需要发出通知,这里就是最基本的通知的写法。发送名为postdata的通知并传送一个数组dataArray。
1 2 | NSMutableArray *dataArray = [[ NSMutableArray alloc] initWithObjects:@ "1" , @ "2" , @ "3" , nil ]; [[ NSNotificationCenter defaultCenter] postNotificationName:@ "postData" object:dataArray]; |
而在接受的页面我们需要增加观察者并接受数组,这时我们的RAC就派上用场了。
1 2 3 4 | [[[ NSNotificationCenter defaultCenter] rac_addObserverForName:@ "postData" object: nil ] subscribeNext:^( NSNotification *notification) { NSLog (@ "%@" , notification.name); NSLog (@ "%@" , notification.object); }]; |
当这个页面监听到名为postdata的通知时他就会执行block中的方法,当然这里的参数改成id x
也是可以的,这里用NSNotification主要是强调它的类型。让我们看看控制台的输出。
1 2 3 4 5 6 | 2016-01-04 20:10:52.274 RACStudyTest[5918:439077] postData 2016-01-04 20:10:52.275 RACStudyTest[5918:439077] ( 1, 2, 3 ) |
可见,notification.object就是我们想要的数组,当然我们也可以传一些model。值得一提的是,RAC中的通知不需要remove observer
,因为在rac_add方法中他已经写了remove。
4.KVO
RAC中得KVO大部分都是宏定义,所以代码异常简洁,简单来说就是RACObserve(TARGET, KEYPATH)
这种形式,TARGET是监听目标,KEYPATH是要观察的属性值,这里举一个很简单的例子,如果UIScrollView滚动则输出success。
1 2 3 4 5 6 7 | UIScrollView *scrolView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, 0, 200, 400)]; scrolView.contentSize = CGSizeMake(200, 800); scrolView.backgroundColor = [UIColor greenColor]; [ self .view addSubview:scrolView]; [RACObserve(scrolView, contentOffset) subscribeNext:^( id x) { NSLog (@ "success" ); }]; |
1.遍历数组
NSArray
*numbers = @[
@1
,
@2
,
@3
,
@4
];
// 这里其实是三步
// 第一步: 把数组转换成集合RACSequence numbers.rac_sequence
// 第二步: 把集合RACSequence转换RACSignal信号类,numbers.rac_sequence.signal
// 第三步: 订阅信号,激活信号,会自动把集合中的所有值,遍历出来。
[numbers.rac_sequence.signal subscribeNext:^(
id
x) {
NSLog
(
@"%@"
,x);
}];
// 2.遍历字典,遍历出来的键值对会包装成RACTuple(元组对象)
NSDictionary
*dict = @{
@"name"
:
@"xmg"
,
@"age"
:
@18
};
[dict.rac_sequence.signal subscribeNext:^(RACTuple *x) {
// 解包元组,会把元组的值,按顺序给参数里面的变量赋值
RACTupleUnpack(
NSString
*key,
NSString
*value) = x;
// 相当于以下写法
// NSString *key = x[0];
// NSString *value = x[1];
NSLog
(
@"%@ %@"
,key,value);
}];
// 3.字典转模型
// 3.1 OC写法
NSString
*filePath = [[
NSBundle
mainBundle] pathForResource:
@"flags.plist"
ofType:
nil
];
NSArray
*dictArr = [
NSArray
arrayWithContentsOfFile:filePath];
NSMutableArray
*items = [
NSMutableArray
array];
for
(
NSDictionary
*dict in dictArr) {
FlagItem *item = [FlagItem flagWithDict:dict];
[items addObject:item];
}
// 3.2 RAC写法
NSString
*filePath = [[
NSBundle
mainBundle] pathForResource:
@"flags.plist"
ofType:
nil
];
NSArray
*dictArr = [
NSArray
arrayWithContentsOfFile:filePath];
NSMutableArray
*flags = [
NSMutableArray
array];
_flags = flags;
// rac_sequence注意点:调用subscribeNext,并不会马上执行nextBlock,而是会等一会。
[dictArr.rac_sequence.signal subscribeNext:^(
id
x) {
// 运用RAC遍历字典,x:字典
FlagItem *item = [FlagItem flagWithDict:x];
[flags addObject:item];
}];
NSLog
(
@"%@"
,
NSStringFromCGRect
([UIScreen mainScreen].bounds));
// 3.3 RAC高级写法:
NSString
*filePath = [[
NSBundle
mainBundle] pathForResource:
@"flags.plist"
ofType:
nil
];
NSArray
*dictArr = [
NSArray
arrayWithContentsOfFile:filePath];
// map:映射的意思,目的:把原始值value映射成一个新值
// array: 把集合转换成数组
// 底层实现:当信号被订阅,会遍历集合中的原始值,映射成新值,并且保存到新的数组里。
NSArray
*flags = [[dictArr.rac_sequence map:^
id
(
id
value) {
return
[FlagItem flagWithDict:value];
}] array];