博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
RAC(reactivecocoa)的使用方法
阅读量:7008 次
发布时间:2019-06-28

本文共 6153 字,大约阅读时间需要 20 分钟。

什么是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

所以我们就可以以这段代码为模板进行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];

 

转载于:https://www.cnblogs.com/Yun-Longcom/p/6632522.html

你可能感兴趣的文章
day5 列表
查看>>
git meld 来merge,diff
查看>>
[BZOJ3754]Tree之最小方差树
查看>>
Spring中不同生命周期Bean的依赖管理
查看>>
项目管理理论与实践(2)——软件需求分析
查看>>
模仿黑魂锁定目标功能
查看>>
领域驱动设计-3-模型的管理
查看>>
第十七课:二阶系统下,正弦稳态
查看>>
第二十五课:CMOS 和能量
查看>>
Ajax本地跨域问题 Cross origin requests are only supported for HTTP
查看>>
SignalR系列续集[系列6:使用自己的连接ID]
查看>>
一步步学习EF Core(3.EF Core2.0路线图)
查看>>
Loj #3057. 「HNOI2019」校园旅行
查看>>
【总结整理】webGIS须知
查看>>
bzoj 1036: [ZJOI2008]树的统计Count——树链剖分
查看>>
类和对象的方法与属性---懒加载与私有事件---单例与私有化构造函数
查看>>
(一)仅仅用ApplicationContext加载界面
查看>>
哈希表的原理及实现代码
查看>>
源辰项目-1
查看>>
react-native app集成阿里云推送
查看>>