ReactiveCocoa(简称为RAC),是由Github开源的一个应用于iOS和OS开发的框架.
ReactiveCocoa 优点大家或多或少都有所了解,不管是 处理业务逻辑 还是 降低耦合度方面都是非常好,我感觉最棒的还是RAC的编程思想,函数响应式编程 ,不需要考虑调用顺序,和太多逻辑,直接考虑结果,把每一次操作都写成一系列嵌套的方法中,使代码高聚合,方便管理。我最近的的写代码方式 也在慢慢被RAC改变。
-
ReactiveCocoa 分为四个库:ReactiveCocoa、ReactiveSwift、ReactiveObjC、ReactiveObjCBridge。
ReactiveObjC 是 RAC的Object-C版本
使用CocoaPods导入
在导入过程中 podfile如果只描述pod 'ReactiveObjC' 会提示有问题,需要在podfile加上use_frameworks!(感叹号也不能少),重新pod install 才能导入成功。
这篇 先写一些常用的 基础方法 方便以后复制粘贴😄,以后计划再写一个小项目 基于RAC+MVVM 到时候在结合项目写一篇
- 数组
//数组遍历
NSArray *array = @[@"1", @"2", @"3", @"4", @"5"];
[array.rac_sequence.signal subscribeNext:^(id _Nullable x) {
NSLog(@"遍历数组 %@",x);
}];
// 数组内容替换 并生成一个新的数组
NSArray *newArray = [[array.rac_sequence mapReplace:@"one"] array];
NSLog(@"newArray %@",newArray);
- 元祖
元祖和OC 的数组差不多
//创建元祖
RACTuple *tuple = [RACTuple tupleWithObjects:@"1", @"2", @"3", @"4", @"5", nil];
// 从数组中获取内容
RACTuple *tuple = [RACTuple tupleWithObjectsFromArray: array];
// 宏创建
RACTuple *tuple = RACTuplePack(@"1", @"2", @"3", @"4", @"5");
NSLog(@"按下标取内容:%@", tuple[2]);
NSLog(@"第一个元素:%@", [tuple first]);
NSLog(@"最后一个元素:%@", [tuple last]);
- 字典
字典遍历
NSDictionary *dict = @{@"name":@"韩梅梅", @"age":@"19",@"lovers":@"╮(╯_╰)╭"};
[dict.rac_sequence.signal subscribeNext:^(RACTuple * _Nullable x) {
NSLog(@"遍历字典 %@",x);
//RACTupleUnpack 是个宏 可快速分开Key value
RACTupleUnpack(NSString *key, NSString *value) = x
NEWSLog(@"字典内容:%@:%@", key, value);
}];
- Button点击事件 代码放在一起,这样非常方便我们管理
UIButton *btn=[UIButton buttonWithType:UIButtonTypeRoundedRect];
btn.frame=CGRectMake(100, 100, 100, 100);
[btn setTitle:@"点击" forState:UIControlStateNormal];
[self.view addSubview:btn];
[[btn rac_signalForControlEvents:UIControlEventTouchUpInside]subscribeNext:^(__kindof UIControl * _Nullable x) {
NSLog(@"😁哈哈哈");
}];
- NSTimer 计时器 ,以往我们使用系统的计时器,不管是 NSTimer 方法还是 GCD 的方法 要不是 要记得及时释放,要不就写一大推。RAC 如此简单就搞定 感觉不要太爽。
//多长时间后执行一次
[[RACScheduler mainThreadScheduler]afterDelay:3 schedule:^{
NSLog(@"3秒后执行一次");
}];
//takeUnti作用是 当控制器消失后取消执行
RACDisposable *dispo=[[[RACSignal interval:2 onScheduler:[RACScheduler mainThreadScheduler]] takeUntil:self.rac_willDeallocSignal ] subscribeNext:^(id x) {
NSLog(@"每两秒执行一次");
}];
[dispo dispose]; //停止定时器
- 代替代理 以往我们控制器上有View 上面有按钮有点击事件,这个需求很常见,如果是用代理完成看看我们需要写什么,
1.View.h 中写 协议, 定义属性
2.View.m 中在方法里调用代理
3.在控制器中遵守协议 设置代理
4.写代理方法
感觉一大推 很麻烦RAC 就在控制器中 用按钮所在的View调用一个方法就搞定。 这属于RAC的发送信号机制的基本表现,注意:调用的方法名字一定要写对
//代替代理
[[self.subView rac_signalForSelector:@selector(btnClick)] subscribeNext:^(RACTuple * _Nullable x) {
NSLog(@" view 中的按钮被点击了");
}];
- 代替代理2
1.给当前控制器添加一个按钮,modal到另一个控制器界面
2.另一个控制器view中有个按钮,点击按钮,通知当前控制器
步骤一:在第二个控制器.h,添加一个RACSubject代替代理。
@interface TwoViewController : UIViewController
@property (nonatomic, strong) RACSubject *delegateSignal;
@end
步骤二:监听第二个控制器按钮点击
@implementation TwoViewController
- (IBAction)notice:(id)sender {
// 通知第一个控制器,告诉它,按钮被点了
// 通知代理
// 判断代理信号是否有值
if (self.delegateSignal) {
// 有值,才需要通知
[self.delegateSignal sendNext:nil];
}
}
@end
步骤三:在第一个控制器中,监听跳转按钮,给第二个控制器的代理信号赋值,并且监听.
@implementation OneViewController
- (IBAction)btnClick:(id)sender {
// 创建第二个控制器
TwoViewController *twoVc = [[TwoViewController alloc] init];
// 设置代理信号
twoVc.delegateSignal = [RACSubject subject];
// 订阅代理信号
[twoVc.delegateSignal subscribeNext:^(id x) {
NSLog(@"点击了通知按钮");
}];
// 跳转到第二个控制器
[self presentViewController:twoVc animated:YES completion:nil];
}
@end
- 代替KVO
[[self rac_valuesForKeyPath:@"name" observer:self] subscribeNext:^(id _Nullable x) {
NSLog(@"name属性的改变:%@", x);
}];
//RACObserve 宏写法
[RACObserve(self, self.name) subscribeNext:^(id _Nullable x) {
NSLog(@"name属性的改变111:%@", x);
}];
- 监听 TextField 的输入改变 告别代理了
监听 TextField 的输入
[[textField rac_textSignal] subscribeNext:^(NSString * _Nullable x) {
NSLog(@"内容:%@", x);
}];
//也可以添加判断条件
[[textField.rac_textSignal filter:^BOOL(NSString * _Nullable value) {
return value.length > 10; // 文字长度 > 10 在监听
}] subscribeNext:^(NSString * _Nullable x) {
NSLog(@"输入框内容:%@", x);
}];
- 监听 Notification 通知事件 以往 还要清除通知 有了RAC 不在那么麻烦
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:NIGHT object:nil] subscribeNext:^(id x) {
NSLog(@"夜间模式:%@",x);
}];
- 手势 代码放在一起,这样非常方便我们管理
UIPanGestureRecognizer *panGesture = [[UIPanGestureRecognizer alloc] init];
[panGesture.rac_gestureSignal subscribeNext:^(id x) {
NSLog(@"拖动手势:%@", x);
}];
[self.view addGestureRecognizer:panGesture];
结尾:水平有限,代码也很烂,一直在努力学习中,大家多多包涵。