RAC用法详解,一看就懂,以及RAC中的循环引用
最近一段时间开始研究RAC,它集响应式编程,函数式编程和链式编程于一身,官方的说,ReactiveCocoa(其简称为RAC)是由GitHub开源的一个应用于iOS和OS X开发的新框架,大大简化了我们的代码,提高了代码的安全性,很多处理已经帮我们做好了,比如block里面循环引用的问题,最终在我们取消订阅的那一刹那,帮我们打断了循环引用。
首先,如何安装RAC,直接通过pods进行安装即可,现在RAC已经支持swift,这里我们就直接基于OC的讲解吧。
推荐RAC的版本3.0.0,通过pod安装好后,build一下看看有没有问题,一般情况式没啥问题的。
现在一起开启RAC的愉快之旅吧。
由于RAC的功能实在是太强大了,涉及到一整套的编程逻辑,比如数据结构方面,增加了元祖类型RACTuple,RACSequence;数据处理方面,快速遍历数组,字典(非常好的字典转模型设计),监听各种事件,文本框输入,代替通知,代理,KVO等等,网络事件的处理,比如一个页面多个请求完成后刷新页面等等,还有非常牛逼的宏(当然我没看懂,不建议大家去看,会用就好了,看了会疯掉的),列举了这么多,其实最根本的就是信号的处理,所有的事件都是先转化成一个信号RACSignal,再订阅信号来完成的。
知识点一:
数据结构,元祖类,RACTuple,比较类似于OC中的数组,用法上也大同小异,可以存放任何类型的对象。
RACTuple *tuple = [RACTuple tupleWithObjectsFromArray:@[@"aaa",@"bbb",@(123)]];
知识点二:快速遍历数组
数据结构,RACSequence,用于代理NSArray,NSDictionary,主要用途是进行快速遍历,最常用的场景就是字典转模型。
NSArray *arr = @[@"abc",@"bbb",@123];
RACSequence *requence = arr.rac_sequence; // 字典或数组都可以直接转化成RACSequence类型的对象
// 遍历
RACSignal *signal = requence.signal; // 再拿到这个对象的信号属性进行遍历
[signal subscribeNext:^(id _Nullable x) {NSLog(@"%@",x);}];
// 快速写法:
[arr.rac_sequence.signal subscribeNext:^(id _Nullable x) {}];
知识点三:快速遍历字典
// 字典
NSDictionary *dict = @{@"name":@"Hank",@"age":@18};
// 字典转集合,遍历,其中的x其实就是元祖类型,拿到元祖RACTuple即可进行拆包处理
[dict.rac_sequence.signal subscribeNext:^(id _Nullable x) {
// NSLog(@"%@",x);
// 发送了元祖类型 RACTwoTuple
// RACTwoTuple *tuple = (RACTwoTuple *)x;
// NSString *key = tuple[0];
// NSString *value = tuple[1];
// 宏,用于解析元祖,参数:需要解析的元祖,拆包去处键值数据
RACTupleUnpack(NSString *key,NSString *value) = x;
NSLog(@"%@%@",key,value);
}];
知识点四:举例应用场景:快速遍历,数据转模型并封装成数组
// 解析plist文件
NSString *filePath = [[NSBundle mainBundle]pathForResource:@"KFC.plist" ofType:nil];
NSArray *dictArray = [[NSArray alloc]initWithContentsOfFile:filePath];
// 字典转模型,解决方案一
// NSMutableArray *array = [[NSMutableArray alloc]init];
// [dictArray.rac_sequence.signal subscribeNext:^(NSDictionary * _Nullable x) {
// KFC *kfc = [KFC KFCWithDict:x];
// [array addObject:kfc];
// }];
// 解决方案二:通过映射,可以知道里面是字典,所以写字典
// 会将集合中的所有元素value映射成新的对象[KFC KFCWithDict:value],然后包装在一个数组里面
NSArray *arr = [[dictArray.rac_sequence map:^id _Nullable(NSDictionary * value) {
// 返回一个模型
return [KFC KFCWithDict:value];
// 转成数组,拿到数据直接用是不是很方便啊!!!
}]array];
NSLog(@"%@",arr);
知识点五: 取代代理,监听某个类的事件
一般我们在权衡数据回调时,会倾向于使用block进行回调,但是对于一些事件的处理,比如点击事件,去重写block或者代理,显得十分麻烦,这里推荐使用RAC来解决。
比如,监听一个类的点击事件或者某个事件的执行,
// 1.代替代理:RACSubject
// 监听blueView里面的哪个方法的调用
[[self.blueView rac_signalForSelector:@selector(laile:)]subscribeNext:^(RACTuple * _Nullable x) {
NSLog(@"dianjile:%@",x);
}];
[[self.blueView rac_signalForSelector:@selector(btnClick:)]subscribeNext:^(RACTuple * _Nullable x) {
NSLog(@"dianjile:%@",x);
}];
}
知识点六:取代KVO
原生的KVO是将监听和监听后的处理分开,并且还要在一个类的dealloc方法里面进行注销观察者,显得非常麻烦,而RAC很好的帮我们处理好了一切,见下面:
// 监听方案一:
[_blueView rac_observeKeyPath:@"frame" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld observer:nil block:^(id value, NSDictionary *change, BOOL causedByDealloc, BOOL affectedOnlyLastComponent) {
// 监听的回调
// 好处:将多个监听分开,也不需要担心观察者的移除
NSLog(@"value:%@---change:%@",value,change);
}];
// 监听方案2:
// 创建一个监听的信号
// [[_blueView rac_valuesForKeyPath:@"frame" observer:nil
// ] subscribeNext:^(id _Nullable x) {
// NSLog(@"%@",x);
// }];
知识点七:监听当前控制器的某个事件:
// 代替Controller的监听事件
[[_btn rac_signalForControlEvents:(UIControlEventTouchUpInside)]subscribeNext:^(__kindof UIControl * _Nullable x) {
NSLog(@"%@",x);
}];
知识点八:取代通知中心
// 代理通知
[[[NSNotificationCenter defaultCenter]rac_addObserverForName:UIKeyboardWillShowNotification object:nil]subscribeNext:^(NSNotification * _Nullable x) {
// 全部转化成了信号
NSLog(@"%@",x);
}];
知识点九:监听文本框的输入
// 监听文本框:拿到文本框输入的每次的值
[_textView.rac_textSignal subscribeNext:^(NSString * _Nullable x) {
NSLog(@"%@",x);
}];
知识点十:RAC的timer计时器
原生的NSTimer用起来比较麻烦, 创建后需要手动的停止,释放,而RAC也很好的帮我们处理好了一切:
// RAC的timer
[[RACSignal interval:1.0 onScheduler:[RACScheduler scheduler]]subscribeNext:^(NSDate * _Nullable x) {
NSLog(@"%@",[NSThread currentThread]);
}];
知识点十一:RAC处理网络请求事件,比如一个页面有多个接口,需要一起刷新UI时。
/ 请求数据 RACSignal *signal1 = [RACSignal createSignal:^RACDisposable * _Nullable(id_Nonnull subscriber) {
// 请求网络数据 NSLog(@"请求网络数据 1");
// 发送数据 [subscriber sendNext:@"发送数据 1"];
return nil;
}];
RACSignal *signal2 = [RACSignal createSignal:^RACDisposable * _Nullable(id_Nonnull subscriber) {
// 请求网络数据
NSLog(@"请求网络数据 2");
// 发送数据
[subscriber sendNext:@"发送数据 2"];
return nil;
}];
// 订阅信号后就会收到信号发出的内容
// [signal1 subscribeNext:^(id _Nullable x) {
// NSLog(@"处理数据 1");
// }];
// 数组:存放信号
// 当数组中所有信号都发送了数据,才会执行selector,必须带参,参数是sendNext发送过来的,必须和信号一一对应
// 参数:每个信号发送的数据,这里必须要带参数,不然会崩溃
[self rac_liftSelector:@selector(updateUIWithOneData:twoData:) withSignalsFromArray:@[signal1,signal2]];
知识点十二:RAC里面的宏
RAC里面的宏可以分为以下几类:
/**
* RAC,给某个对象绑定一个属性
* RACObserver,监听某个对象的属性
* RACTuplePack:将数据打包成RACTuple
* RACTupleUnpack: 将RAC元祖解包成数据
*/
比如RAC宏用法:
// 监听文本框内容
// [self.textfield.rac_textSignal subscribeNext:^(NSString * _Nullable x) {
// _label.text = x;
// }];
// 宏RAC,给某个对象的某个属性绑定信号,一旦信号产生数据,就会将内容赋值给属性
RAC(_label,text) = self.textfield.rac_textSignal;
// 监听某个对象的某个属性,返回信号
// 只要这个对象的属性发生变化,信号就发送数据
[RACObserve(self.label, text)subscribeNext:^(id _Nullable x) {
NSLog(@"%@",x);
}];