ReactiveCocoa入门教程:第三部分

什么是RAC?

几乎每一篇介绍RAC的文章开头都是这么一个问题。我这篇文章是写给新手(包括我自己)看的,所以这个问题更是无法忽视。

简单的说,RAC就是一个第三方库,他可以大大简化你的代码过程。

官方的说,ReactiveCocoa(其简称为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 <ReactiveCocoa/ReactiveCocoa.h>

接下来就是最关键的RAC代码了。

[[self.textFild rac_signalForControlEvents:UIControlEventEditingChanged] subscribeNext:^(id x){ NSLog(@"change"); }];

就这么短短的两行代码。他实现了一个功能,即监听了textFild的UIControlEventEditingChanged事件,当事件发生时实现方法NSLog
所以我们就可以以这段代码为模板进行RAC的使用,举一反三,以后的UIButton点击事件我们都可以用RAC方法进行添加,再也不用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];

猜你喜欢

转载自blog.csdn.net/qq_26918391/article/details/77448690