ReactiveCocoa方法总结
2022-01-07 本文已影响0人
林希品
基础部分
/*************************************基础部分**************************************************/
#pragma mark --textField属性变化
-(void)textFieldChange
{
//初始化一个textField控件
UITextField *textField = ({
UITextField *textField = [[UITextField alloc] init];
textField.backgroundColor = [UIColor cyanColor];
textField ;
});
textField.delegate = self ;
[self.view addSubview:textField];
[textField mas_makeConstraints:^(MASConstraintMaker *make) {
make.size.mas_equalTo(CGSizeMake(180, 40));
make.center.equalTo(self.view);
}];
/*****************监听textField的属性变化情况*******************/
//RAC内部封装好的类
//默认执行1次,所以会有打印,原因:请查看RAC底层bind,hook的思想
//监听文本框的文字改变
[textField.rac_textSignal subscribeNext:^(id x) {
NSLog(x);
}];
}
#pragma mark -- Button用法 监听事件
-(void)textBtn
{
UIButton *button = [UIButton buttonWithType:UIButtonTypeSystem];
[button setTitle:@"点击" forState:UIControlStateNormal];
button.backgroundColor = [UIColor redColor];
[self.view addSubview:button];
@weakify(self);
[button mas_makeConstraints:^(MASConstraintMaker *make) {
@strongify(self);
make.size.mas_equalTo(CGSizeMake(self.view.bounds.size.width, 30));
make.bottom.equalTo(self.view).offset(0);
}];
// 监听事件
// 把按钮点击事件转换为信号,点击按钮,就会发送信号
[[button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
LxPrintf(@"进来了");
RACViewController *VC = [[RACViewController alloc] init];
@strongify(self);
[self presentViewController:VC animated:YES completion:^{
LxPrintf(@"%s",__func__);
}];
}];
}
#pragma mark --手势
-(void)addTap
{
/**************手势*****************/
self.view.userInteractionEnabled = YES ;
UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc] init];
[[tap rac_gestureSignal] subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
[self.view addGestureRecognizer:tap];
}
#pragma mark --通知中心
-(void)notifiCenter
{
/**************通知*****************/
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationDidEnterBackgroundNotification object:nil] subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
}
#pragma mark --代理
-(void)delegateDemo
{
UIAlertView *alterView = [[UIAlertView alloc] initWithTitle:@"RAC" message:@"ReactiveCocoa" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Ensure", nil];
//方法一:代理
// [[self rac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)] subscribeNext:^(id x) {
//
// RACTuple *tuple = (RACTuple *)x ; //类似于swift的元组
// LxDBAnyVar(tuple);
// LxDBAnyVar(tuple.first);
// LxDBAnyVar(tuple.second);
// LxDBAnyVar(tuple.third);
// }];
[alterView show];
//方法二:内部封装的方法
//更简单的方式
[[alterView rac_buttonClickedSignal] subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
}
#pragma mark -- KVO
-(void)addKvo
{
UIScrollView *scrollView = [[UIScrollView alloc] init];
scrollView.delegate = (id<UIScrollViewDelegate>)self ;
[self.view addSubview:scrollView];
UIView *scrollViewContentView = [[UIView alloc] init];
scrollViewContentView.backgroundColor = [UIColor yellowColor];
[scrollView addSubview:scrollViewContentView];
@weakify(self);
[scrollView mas_makeConstraints:^(MASConstraintMaker *make) {
@strongify(self);
make.edges.equalTo(self.view).insets(UIEdgeInsetsMake(80, 80, 80, 80));
}];
[scrollViewContentView mas_makeConstraints:^(MASConstraintMaker *make) {
@strongify(self);
make.edges.equalTo(scrollView);
make.size.mas_equalTo(CGRectGetWidth(self.view.frame), CGRectGetHeight(self.view.frame));
}];
[RACObserve(scrollView, contentOffset) subscribeNext:^(id x) {
LxDBAnyVar(x);
}];
}
//处理当界面有多次请求时,需要都获取到数据时,才能展示界面
-(void)moreRequest
{
/*
1.rac_liftSelector:withSignalsFromArray:Signals:当传入的Signals(信号数组),每一个signal都至少sendNext过一次,就会去触发第一个selector参数的方法。
2.使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
*/
//处理多个请求,都返回结果的时候,统一做处理.
RACSignal *request1 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
// 发送请求1
[subscriber sendNext:@"发送请求1"];
return nil;
}];
RACSignal *request2 = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
// 发送请求2
[subscriber sendNext:@"发送请求2"];
return nil;
}];
// 使用注意:几个信号,参数一的方法就几个参数,每个参数对应信号发出的数据。
[self rac_liftSelector:@selector(updateUIWithR1:r2:) withSignalsFromArray:@[request1,request2]];
}
// 更新UI
- (void)updateUIWithR1:(id)data r2:(id)data1
{
NSLog(@"更新UI%@ %@",data,data1);
}
#pragma mark --定时器
-(void)RACSchedulerAndMain
{
/**************定时器*****************/
//1.延迟某个时间再做某件事
// [[RACScheduler mainThreadScheduler] afterDelay:2 schedule:^{
// LxDBAnyVar(rac);
// }];
//2.每间隔多长时间做一件
[[RACSignal interval:1 onScheduler:[RACScheduler mainThreadScheduler]] subscribeNext:^(id x) {
LxDBAnyVar(x);
// LxPrintAnything(@"55555555");
}];
//这是定时器最常用的两种方法,第一种方法,延迟时间去做某件事,更改afterDelay的属性,第二种方法,每间隔躲藏时间去做一件事,更改interval的属性
}
1、RACSubject当做代理使用
步骤一:在第二个控制器.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
2、RACSubject遍历 map处理
1.遍历数组
NSArray *numbers = @[@1,@2,@3,@4];
// 这里其实是三步
// 第一步: 把数组转换成集合RACSequence numbers.rac_sequence
// 第二步: 把集合RACSequence转换RACSignal信号类,numbers.rac_sequence.signal
// 第三步: 订阅信号,激活信号,会自动把集合中的所有值,遍历出来。
[numbers.rac_sequence.signal subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
2.遍历字典,遍历出来的键值对会包装成RACTuple(元组对象)
NSDictionary *dict = @{@"name":@"xmg",@"age":@18};
[dict.rac_sequence.signal subscribeNext:^(RACTuple *x) {
// 解包元组,会把元组的值,按顺序给参数里面的变量赋值
RACTupleUnpack(NSString *key,NSString *value) = x;
// 相当于以下写法
// NSString *key = x[0];
// NSString *value = x[1];
NSLog(@"%@ %@",key,value);
}];
3.字典转模型
3.1 OC写法
NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test.plist" ofType:nil];
NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath];
NSMutableArray *items = [NSMutableArray array];
for (NSDictionary *dict in dictArr) {
TestModel *item = [TestModel yy_modelWithDictionary:dict];
[items addObject:item];
}
3.2 RAC写法
NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test.plist" ofType:nil];
NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath];
NSMutableArray *flags = [NSMutableArray array];
// rac_sequence注意点:调用subscribeNext,并不会马上执行nextBlock,而是会等一会。
[dictArr.rac_sequence.signal subscribeNext:^(id x) {
// 运用RAC遍历字典,x:字典
TestModel *item = [TestModel yy_modelWithDictionary:dict];
[flags addObject:item];
}];
3.3 RAC高级写法:
NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test.plist" ofType:nil];
NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath];
// map:映射的意思,目的:把原始值value映射成一个新值
// array: 把集合转换成数组
// 底层实现:当信号被订阅,会遍历集合中的原始值,映射成新值,并且保存到新的数组里。
NSArray *flags = [[dictArr.rac_sequence map:^id(id value) {
return [TestModel yy_modelWithDictionary: value];
}] array];
小白遍历 map学习处理
小白遍历 map学习处理
小白遍历 map学习处理
NSArray *numberTwo = @[@(123),@(123),@(123),@(123)];
NSLog(@"before = number = %@",numberTwo);
NSArray *mapArray = [[numberTwo.rac_sequence map:^id(id value) {
int changeValue = [value intValue];
return [NSString stringWithFormat:@"%d",changeValue +4];
}] array];
NSLog(@"affter = number = %@",mapArray);
其余常见用法
1 代替KVO :
rac_valuesAndChangesForKeyPath:用于监听某个对象的属性改变。
// 把监听redV的center属性改变转换成信号,只要值改变就会发送信号
// observer:可以传入nil
[[redV rac_valuesAndChangesForKeyPath:@"center" options:NSKeyValueObservingOptionNew observer:nil] subscribeNext:^(id x) {
NSLog(@"%@",x);
}];
2 监听事件:
rac_signalForControlEvents:用于监听某个事件。
// 把按钮点击事件转换为信号,点击按钮,就会发送信号
[[self.btn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
NSLog(@"按钮被点击了");
}];
3 代替通知:
rac_addObserverForName:用于监听某个通知。
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(id x) {
NSLog(@"键盘弹出");
}];
4 监听文本框文字改变:
rac_textSignal:只要文本框发出改变就会发出这个信号。
[_textField.rac_textSignal subscribeNext:^(id x) {
NSLog(@"文字改变了%@",x);
}];
[[self.testTextField.rac_textSignal map: ^id _Nullable(NSString* value) {
return [NSString stringloithFormat:@"map自定义了返回信号:%0",value ]; //改变文字输出
}] subscribeNext:^(id _Nullable x) {
NSLog(@"%@", x);
}];
5.代理
第一种:代替代理
UIAlertView *alterView = [[UIAlertView alloc] initWithTitle:@"RAC" message:@"ReactiveCocoa" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Ensure", nil];
//以UIAlertViewDelegate为例用rac_signalForSelector
//实现- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex代理方法
[[self rac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)] subscribeNext:^(RACTuple * _Nullable tup) {
if(tup){
RACTupleUnpack(UIAlertView *alert,NSNumber *buttonIndex) = tup;
NSInteger seIndex = [buttonIndex integerValue];
NSString *btnTitle = [alert buttonTitleAtIndex:seIndex];
if(alert.tag == 1000){
NSLog(@"btnTitle = %@",btnTitle);
}
}
}
}];
;
[alterView show];
第二种:监听某个对象调用某个方法
// RAC方法:可以判断某个方法有没有调用
// 只要self调用Selector就会产生一个信号
// rac_signalForSelector:监听某个对象调用某个方法
[[self rac_signalForSelector:@selector(loadAdvertisementsData)] subscribeNext:^(id x) {
NSLog(@"控制器调用了loadAdvertisementsData");
}];
// 判断下_btn有没有调用btnClick,有信号就表示点击了按钮
[[_btn rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(id x) {
NSLog(@"点击了按钮");
}];