前言
前段时间学习ReactiveCocoa基础,总结出来一些常用的方法,希望对学习ReactiveCocoa人有所帮助.
ReactiveCocoa简介
- ReactiveCocoa(简称为RAC),是由Github开源的一个应用于iOS和OS开发的新框架.github地址.
1.ReactiveCocoa的思想
- 函数式编程思想:是把操作尽量写成一系列嵌套的函数或者方法调用。
- 函数式编程特点:每个方法必须有返回值(本身对象),把函数或者Block当做参数,block参数(需要操作的值)block返回值(操作结果)
2.ReactiveCocoa常见的类
- RACSiganl:信号类,在RAC中最核心的类,他本身不具备发送信号的能力
- RACSiganl简单的应用:
#pragma mark - RACSignal
- (void)RACSignal {
// RACSignal 有数据产生的时候 就会使用RACSignal RACSignal是不能发送信号的
//RACSignal使用步骤: 1.创建信号(冷信号) 2.订阅信号(热信号) 3.发送信号
//RACSignal使用步骤
//1.创建信号(冷信号)
RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
//didsubscriber调用:只要一个信号被订阅就会调用
//didsubscriber作用:发送数据
//3.发送信号
[subscriber sendNext:@1];
[subscriber sendCompleted];
return nil;
}];
//2.订阅信号(热信号)
[signal subscribeNext:^(id x) {
//nextBlock调用:只要订阅者发送数据就会调用
//nextBlock作用:处理数据 展示到UI界面
// x 信号发送的内容
NSLog(@"%@",x);
}];
}
- RACDisposable:用于取消订阅或者清理资源,信号发送完成或者发送失败的时候会触发他.
- RACDisposable简单的应用:
#pragma mark - RACDisposable
- (void)RACDisposable {
//1.创建信号
RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
//3.发送信号
[subscriber sendNext:@"123"];
// [subscriber sendCompleted];
return [RACDisposable disposableWithBlock:^{
//只要我的信号 取消订阅就会来这
//清空资源
NSLog(@"信号取消订阅");
}];
}];
//2.订阅信号
RACDisposable *disposable = [signal subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
//默认一个信号发送数据完毕就会主动取消订阅
//只要订阅者在 就不会自动取消信号订阅
//取消订阅
[disposable dispose];
}
- RACSubject:信号提供者,自己可以充当信号,又能发送信号,通常代替代理.
- RACSubject:简单的应用:
- 在使用RACSubject时候,如果进行map操作,一定要发送完成信号,不然会内存泄露
#pragma mark - RACSubject
- (void)RACSubject {
//1.创建信号
RACSubject *subject = [RACSubject subject];
//2.订阅信号
//不同信号订阅的方式不一样
//RACSubject处理订阅:仅仅是保存订阅者
[subject subscribeNext:^(id x) {
NSLog(@"接收到信号");
}];
//3.发送数据
[subject sendNext:@1];
//底层实现:遍历所有的订阅者,调用nextBlock
//RACSubject被订阅,仅仅是保存订阅者
//执行流程
}
- RACSubject替换代理
// 需求:
// 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
RACCommand是RAC很重要的组成部分,通常用来表示某个action的执行。
使用场景: 监听按钮点击 网络请求
RACCommand的简单的应用:
#pragma mark - RACCommand
- (void)RACCommand {
//RACCommand: 处理事件
//1.创建命令
RACCommand *command = [[RACCommand alloc]initWithSignalBlock:^RACSignal *(id input) {
//input: 执行命令传入的参数
//Block: 执行命令的时候就会调用
return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
//发送数据
[subscriber sendNext:@"发送数据"];
[subscriber sendCompleted];
return nil;
}];
}];
//如何拿到执行命令中产生的数据
//订阅命令内部的信号
// //执行命令
// RACSignal *signal = [command execute:@1];
// //1.方式一 直接订阅执行命令返回的信号
// [signal subscribeNext:^(id x) {
// NSLog(@"%@",x);
// }];
//1.方式二 executionSignals: 信号源 信号中的信号 executionSignals:信号:发送数据就是信号
//注意:必须要执行命令订阅
// [command.executionSignals subscribeNext:^(RACSignal *x) {
//
// [x subscribeNext:^(id x) {
// NSLog(@"%@",x);
// }];
// }];
//switchToLatest获取最新发送的信号,只能用于信号中信号
[command.executionSignals.switchToLatest subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
[command execute:@1];
}
3.ReactiveCocoa开发中简单应用
- 代替代理 rac_signalForSelector不需要传值 (和RACSubject类似,RACSubject需要传值)
[[_redview rac_signalForSelector:@selector(btnredclick:)] subscribeNext:^(id x) {
NSLog(@"控制器知道按钮被点击");
}];
- 监听事件
//_btn UIButton
[[_btn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
NSLog(@"按钮被点击了"); }];
- 代替通知
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(id x) {
NSLog(@"%@",x); }];
- 监听文本框
// _textField UITextField
[[_textField rac_textSignal] subscribeNext:^(id x) {
NSLog(@"%@",x); }];
- RACSequence遍历数组
NSArray *array = @[@"67",@"45",@"12"];
//array.rac_sequence转换成集合
//array.rac_sequence.signal集合转化成信号
//subscribeNext 订阅信号 内部会自动遍历所有的元素法发出来
[array.rac_sequence.signal subscribeNext:^(id x) {
NSLog(@"%@",x); }];
- RACSequence遍历字典
NSDictionary *dict = @{@"niming":@"小明",@"ok":@"好的"};
[dict.rac_sequence.signal subscribeNext:^(RACTuple *x) {
//RACTuplePack:用来解析元组
//宏里面的参数 传需要解析出来的变量名
// = 右边,放需要解析的元组
RACTupleUnpack(NSString *key,NSString *value) = x;
NSLog(@"%@ %@",key,value);
}];
- 当一个界面有多次请求时候, 需要保证全部请求完成,才能搭建界面
//数组: 存放信号
//数组中的所有信号都发送数据时候, 才会执行Selector
//方法的参数:必须跟数组的信号--对应
//方法的参数就是每一个信号发送的数据
[self rac_liftSelector:@selector(upDateUIwithHotData:newData:) withSignalsFromArray:@[signal,newSignal]];
RAC常用的宏
- RAC(<#TARGET, ...#>) 用来给某个对象的某个属性绑定信号,只要产生信号内容,就会给内容给属性赋值
基础用法:
RAC(_label,text) = _textField.rac_textSignal;
等同于
//监听文本框的内容
[_textField.rac_textSignal subscribeNext:^(id x) {
_label.text = x;
}];
- @weakify(Obj) @strongify(Obj) 一般两个都是配套使用,解决循环引用问题
- RACObserve(self.view, frame) 监听某个对象的某个属性,返回的是信号
- RACObserve使用时候需要注意循环引用问题 使用@weakify(Obj) @strongify(Obj)
- RACTuplePack 把数据包装成RACTuple (元组类)
基本用法:
//包装元组
RACTuple *tupe = RACTuplePack(@1,@2);
总结
在学习中参考了这篇文章 学习中自己写了两个demo登录界面和网络请求(写了一个UITableView展示数据)整理下传到github.
登录界面代码github
其他学习RAC文章
ReactiveCocoa入门教程
ReactiveCocoa