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);

}];

你可能感兴趣的:(RAC用法详解,一看就懂,以及RAC中的循环引用)