RAC(ReactiveCocoa) 使用详解
RAC 是什么?
- ReactiveCocoa(RAC) github 一个开源框架
- 函数响应式编程框架
RAC 作用
- 面向信号编程
- 帮我们简单霸气的处理事件
RAC 常见类:
-
RACSignal: 信号类
- 通过 RACSignal 创建一个信号(默认:冷信号)
- 通过订阅者,订阅信号(变成:热信号)
- 发送信号
RACDisposable: 它可以帮组我们取消订阅.信号发送完毕失败了.
RACSubscriber( 协议):订阅者(发送信号)
RACSubject:信号提供者,自己可以充当信号,又可以发送信号
一个信号的使用的流程图( RACSignal )
- 创建一个信号(RACDynamicSignal)->保存block(didSubscriber)
- 订阅信号->创建订阅者( RACSubscriber ) -> 保存Block(nextBlock)
- 真正订阅信号->执行didSubscriber->发送数据->执行 nextBlock
扩展
Block 常见的使用
- 作为属性
- 作为参数
- 作为返回值(扩展性非常强)
编程思想:
-
链式编程思想的特点:方法的返回值必须要有方法调用者!(masonry 框架)
- masonry(mas_makeConstraints执行流程)
- 创建约束制造者 MASConstraintMaker, 并且绑定控件,生成一个所有约束的数组
- 执行 mas_makeConstraints 传入的Block
- 让约束制造者安装约束(清空之前的所以约束,遍历数组一个一个安装)
-
响应式编程:(一响应,就变化)
- KVO(KVO 底层实现)(Person 类为例)
- 自定义Person子类
- 重写监听的属性的 set(setName:)方法,在内部恢复父类做法,通知观察者
- 如何让外界调用自定义 Person 的子类方法,修改当前对象的 isa 指针,指向自定义的子类.
函数式编程思想:是将操作尽可能的写在一起,嵌套函数!本质:就是在方法里传入 Block, 方法中嵌套 Block 调用
RAC 常见的使用方法和函数
- tip1: RACSignal 的使用
// RACSignal: 信号类,当我们有数据产生,创建一个信号!
//1.创建信号(冷信号!)
//didSubscribe调用:只要一个信号被订阅就会调用!!
//didSubscribe作用:利用subscriber发送数据!!
RACSignal * signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
//3.发送数据subscriber它来发送
[subscriber sendNext:@"呵呵哈哈嘿嘿"];
return nil;
}];
//2.订阅信号(热信号!!)
//nextBlock调用:只要订阅者发送数据就会调用!
//nextBlock作用:处理数据,展示UI界面!
[signal subscribeNext:^(id x) {
//x:信号发送的内容!!
NSLog(@"%@",x);
}];
- tip2: RACDisposable 的使用
//2.订阅信号(热信号!!)
RACDisposable * disposable = [signal subscribeNext:^(id x) {
//x:信号发送的内容!!
NSLog(@"%@",x);
}];
//默认一个信号发送数据完毕就会主动取消订阅
//只要订阅者在,就不会自动取消订阅
//手动取消订阅
[disposable dispose];
- tip3: RACSubject 的使用
//1.创建信号
RACSubject * subject = [RACSubject subject];
//2.订阅信号
//不同的信号订阅的方式不一样!!(因为类型不一样,所以调用的方法不一样)
//RACSubject处理订阅:拿到之前的_subscribers保存订阅者
[subject subscribeNext:^(id x) {
NSLog(@"接受到了数据:%@",x);
}];
//3.发送数据
//遍历出所有的订阅者,调用nextBlock
[subject sendNext:@"汉堡包"];
RAC 中的集合类
- 元祖(RACTuple)
- RACSequence
实例:
tip1
//数组
NSArray * arr = @[@"abc",@"bbb",@123];
//RAC集合
// RACSequence * requence = [arr rac_sequence];
// //遍历
// RACSignal * signal = [requence signal];
//
// //订阅信号
// [signal subscribeNext:^(id _Nullable x) {
// NSLog(@"%@",x);
// }];
[arr.rac_sequence.signal subscribeNext:^(id _Nullable x) {
NSLog(@"%@",x);
}];
tip2:
//字典
NSDictionary * dict = @{@"name":@"hank",@"age":@"18"};
//字典转集合
[dict.rac_sequence.signal subscribeNext:^(RACTuple* x) {
// NSString * key = x[0];
// NSString * value = x[1];
// NSLog(@"%@%@",key,value);
//RACTupleUnpack:用来解析元祖
//宏里面的参数,传需要解析出来的变量名称
// = 右边,放需要解析的元祖
RACTupleUnpack(NSString * key,NSString * value) = x;
NSLog(@"%@ : %@",key,value);
}];
tip3:
//会将一个集合中的所有元素都映射成一个新的对象!
NSArray * arr = [[dictArr.rac_sequence map:^id _Nullable(NSDictionary * value) {
//返回模型!!
return [KFC kfcWithDict:value];
}] array];
NSLog(@"%@",arr);
RAC 常见用法
- 代理代理
- 代理 KVO
- 监听 UIControl 的事件
- 代替通知
- 监听文本框的变化
- timer 事件(不用交给 runloop 不会卡死主线程的 UI 操作)
RAC 中常见的宏
- RAC: 给某个对象绑定一个属性!
- RACObserve: 监听某个对象的某个属性!
- RACTuplePack: 打包成一个 RACTuple
- RACTupleUnpack:将 RAC 元祖解包成对应的数据
RAC 中的映射
- flattenMap - 处理发送过来的Value,包装修改过的内容发送出去
- map - 返回的数据就是需要处理的数据
RAC 中的组合
- zip - 两个信号压缩,只有当两个信号同时发送的信号内容,并且将内容合并成为一个元祖给你
- merge - 根据发送的情况发送数据
- then - 忽略第一个信号的所有值
- concat - 按照顺序组合
RAC 中的过滤
- filter - Value(原信号内容),满足条件返回新的信号返回
- ignore - 忽略(忽略指定的值 Value)
- take - 指定拿到前面的哪几条数据(从前向后)
- takeLast - 指定拿到前面的哪几条数据(从后向前) 主意:一定要写结束!
- distinctUntileChanged - 忽略重复的数据
- skip - 跳跃几个值
MVC - 逻辑清晰
MVVM - M 模型 V 视图+控制器 VM:视图模型(核心就是给 C 瘦身,代理控制器做逻辑处理)