ReactivieCocoa的底层就不说明了。这里就简单介绍它的几个基本使用场景。
Cocoapods导入ReactiveCocoa5.0以上版本注意事项
1、如果你使用的是纯swift工程,podfile文件描述如下:
use_frameworks! target '工程名称' do pod 'ReactiveCocoa', '5.0.0-alpha.3' end
2、如果你使用的是纯oc工程,podfile文件描述如下:
use_frameworks! target '工程名称' do pod 'ReactiveObjC', '~> 2.1.0' end
3、如果你使用oc与swift混合编程,podfile文件描述如下:
use_frameworks! target '工程名称' do pod 'ReactiveObjC', '~> 2.1.0' pod 'ReactiveCocoa', '5.0.0-alpha.3' end
基本使用
#import <ReactiveObjC/ReactiveObjC.h>
一、替代代理,代码就是这么的简单,如果传递多个值的话 x是一个RACTuple(元组)对象,这个对象在下面说明,如果值传一个值,那么传的是什么对象,x就是什么对象。
//redView 是在控制器view的子视图,在redView中响应一个事件,把需要的消息从redView传递到控制器过来 //这段代码是在控制器viewController.m中订阅redView的btnClick事件消息 //如果传递多个值的话 x是一个RACTuple(元组)对象,比如@selector(clickWithValue1:value2:) [[_redVeiw rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(id _Nullable x) { NSLog(@"x:%@",x); }];
二、代替KVO
/** 用法和原生代码基本一致valuesForKeyPath: observer: self 是viewController age 是self的一个属性 @param x 是age改变后的值 */ [[self rac_valuesForKeyPath:@"age" observer:nil] subscribeNext:^(id _Nullable x) { NSLog(@"x ::%@",x); }];
三、事件监听
//_btn 是一个按钮,给按钮添加一个事件。不需要像原生那样addTarget:action:forControlEvents:,然后再写一个方法给action这么麻烦了。 [[_btn rac_signalForControlEvents:UIControlEventTouchUpInside]subscribeNext:^(__kindof UIControl * _Nullable x) { NSLog(@"x:::%@",x); }];
四、代替通知
//这个用法就和系统一样了,只是把监听处理的代码聚合在一起,不需要另外写一个方法,提高代码阅读性。 //我这里是监听了一个键盘弹起的一个通知。 [[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(NSNotification * _Nullable x) { NSLog(@"键盘起来了:%@",x); }];
五、输入框改变监听
//这个使用方式和给按钮添加事件监听差不多。这里ControlEvents不需要填了,监听textField的text的改变 [_textfield.rac_textSignal subscribeNext:^(NSString * _Nullable x) { NSLog(@"textfield改变了:%@",x); }];
六、处理几个信号完成后统一处理事件。 比如说:我们在一个界面有几个不同的网络数据请求,我们需要监听他们都请求完成后才处理一些事情,那么就可以使用以下方式。
//创建信号1 RACSignal * signalA = [RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber> _Nonnull subscriber) { //处理信号 。。 正在发送请求1.。 /* 网络请求代码。。。 */ //请求完成后,发送数据 [subscriber sendNext:@"请求1完成"]; return nil; }]; //创建信号2 RACSignal * signalB = [RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber> _Nonnull subscriber) { //处理信号 。。 正在发送请求2.。 /* 网络请求代码。。。 */ //请求完成后,发送数据 [subscriber sendNext:@"请求2完成"]; return nil; }]; //监听 signalA 和 signalB 两个信号都完成了才调用request1WithContent:request2WithContent: //注意request1WithContent:request2WithContent:是自己写的一个方法,但是这个方法有个原则,就是有几个信号,就对应传递几个值,不可以多,也不可以少,否则运行的时候就会崩溃,报方法错误(参数错误) [self rac_liftSelector:@selector(request1WithContent:request2WithContent:) withSignalsFromArray:@[signalA,signalB]];
/** 请求1和请求都完成了 */ -(void)request1WithContent:(NSString *)content1 request2WithContent:(NSString *)content2{ NSLog(@"请求1内容:%@",content1); NSLog(@"请求2内容:%@",content2); }
七、RACTuple元组对象,在第一步代替代理的时候提及过。元组对象和我们的数组对象差不多,使用方法也比较简单,RAC提供了几个宏来创建以及解包。
/** 元组 */ RACTuple * tuple = RACTuplePack(@1,@2,@3);//使用RACTuplePack宏来快速创建 NSLog(@"TUPLE:%@",tuple); //使用RACTupleUnpack宏快速解包 RACTupleUnpack(NSNumber *num1,NSNumber *num2,NSNumber *num3) = tuple; NSLog(@"num1:%@",num1); NSLog(@"num2:%@",num2); NSLog(@"num3:%@",num3); //使用下标的方式来获取 NSLog(@"第0个:%@",tuple[0]);
上面介绍了 RACTuplePack 和 RACTupleUnpack 两个宏的使用,下面再介绍几个常用的宏:
1、RAC 绑定一个信号
//1、RAC 把一个对象的摸个属性绑定一个信号,只有发出信号,就会吧信号的内容给对象的属性赋值。 //这里吧label的text属性绑定到textField改变信号中,textfield的内容发生改变的时候就会发出信号,label的text就会跟随着改变。 RAC(self.label,text) = _textfield.rac_textSignal;
2、RACObserve 相当于kvo使用
//2、KVO /** @param self 是viewController @param age self的一个属性 */ [RACObserve(self, age) subscribeNext:^(id _Nullable x) { NSLog(@"x:%@",x); }];
3、@weakify 和@strongify 这个就是为了解决在使用block的时候强引用问题,注意这两个宏是配套使用才有效。
//3、把一个对象转换成弱指针 @weakify(self);//self 是viewController 在block外面使用@weakify self.signal = [RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber> _Nonnull subscriber) { //在block里面再使用@strongify @strongify(self); NSLog(@"%@,",self.view); return nil; }];