ReactiveObjC使用 ReactiveObjC使用
ReactiveObjC使用
ReactiveCocoa是Github开源的一个用于iOS和OS开发的新框架,Cocoa是苹果整套框架的简称。敢自称为XXXCocoa框架可以想象到这个框架的牛逼!
现在分为ReactiveObjC和ReactiveSwift,两个框架的功能使用相似,本文主要介绍ReactiveObjC的简单使用,希望能对你有所帮助......
ReactiveObjC框架的简单使用
如果是第一次使用,建议先简单测试通过后,再使用...... (你懂的)ReactiveObjC框架的简单介绍
#pragma mark -- 监听事件(按钮点击)
原理:将系统的UIControlEventTouchUpInside事件转化为信号、我们只需要订阅该信号就可以了。
点击按钮的时候触发UIControlEventTouchUpInside事件---> 发出信号 实际是: 执行订阅者(subscriber)的sendNext方法
// 外界使用 [[self.button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(__kindof UIControl * _Nullable x) { //x 就是被点击的按钮 NSLog(@"按钮被点击了%@", x); }];
图例:
#pragma mark -- 代替代理
需求:自定义redView,监听红色view中按钮点击
之前都是需要通过代理监听,给红色View添加一个代理属性,点击按钮的时候,通知代理做事情,符合封装的思想。
rac_signalForSelector:把调用某个对象的方法的信息转换成信号(RACSubject),就会调用这个方法,就会发送信号。
这里表示只要监听了redView的btnClick:方法。(只要redView的btnClick:方法执行了,就会执行下面的方法,并且将参数传递过来)
[[redView rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(id x) { NSLog(@"点击红色视图中的按钮", x); }];
#pragma mark -- 代替KVO
// 把监听redView的center属性改变转换成信号,只要值改变就会发送信号
// observer:可以传入nil
[[redView rac_valuesAndChangesForKeyPath:@"center" options:NSKeyValueObservingOptionNew observer:nil] subscribeNext:^(id x) { NSLog(@"%@",x); }];
#pragma mark -- 代替通知
// 把监听到的通知转换信号
[[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] takeUntil:[self rac_willDeallocSignal]] subscribeNext:^(NSNotification * _Nullable x) { NSLog(@"%@", x); }];
#pragma mark -- 监听文本框的文字改变
// 监听文本框的文字改变、获取文本框文字改变的信号
[_textField.rac_textSignal subscribeNext:^(id x) { self.textLabel.text = x; NSLog(@"文字改变了%@",x); }];
#pragma mark -- 处理多个请求
// 处理多个请求,都返回结果的时候,统一做处理.
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;
}];
// 使用注意:几个信号,selector的方法就几个参数,每个参数对应信号发出的数据。
不需要订阅:不需要主动订阅,内部会主动订阅
[self rac_liftSelector:@selector(updateUIWithR1:r2:) withSignalsFromArray:@[request1,request2]];
// 更新UI
- (void)updateUIWithR1:(id)data r2:(id)data1
{
NSLog(@"更新UI%@ %@",data,data1);
}
#pragma mark -- 遍历数组
// 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); }];
***********************************************RAC常用宏*****************************************
#pragma mark--KVO
RACObserve(就是一个宏定义):快速的监听某个对象的某个属性改变
监听self.view的center属性,当center发生改变的时候就会触发NSLog方法
[RACObserve(self.view, center) subscribeNext:^(id x) { NSLog(@"%@", x); }];
#pragma mark--登录按钮的状态实时监听
RAC(_loginButton, enabled) = [RACSignal combineLatest:@[_username.rac_textSignal, _password.rac_textSignal] reduce:^id _Nullable(NSString * username, NSString * password){ return @(username.length && password.length); }];
#pragma mark--循环引用
// @weakify() 宏定义
@weakify(self) //相当于__weak typeof(self) weakSelf = self;
RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { @strongify(self) //相当于__strong typeof(weakSelf) strongSelf = weakSelf; NSLog(@"%@",self.view); return nil; }]; _signal = signal;
实际开发遇到的坑
一: "引用循环"是肯定会出现的,因此一定要避免"强引用循环"
解决方案:一端使用strong一端使用weak
二: 出现下面的错误:
解决办法:所有控件在使用RAC之前一定要先初始化!先初始化!先初始化!
(Masonry框架:布局之前一定要添加到父控件中)有相似之处