RAC常见用法
1.代替代理
我们还是通过一个例子在了解一下吧
现在呢在我们的控制器中有一个我们自定义的View叫BlueView,里面有一个按钮,点击这个按钮,要我们的控制器去处理事件。
以往呢我们基本就是通过 代理、通知、Block,那么我们用RAC来如何实现呢?
首先在我们的BlueView有一个按钮的点击事件处理方法。
-(IBAction)btnClick:(id)sender;
接下来在我们的控制器中
[[_blueView rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(RACTuple * _Nullable x) {
NSLog(@"%@",x);
}];
rac_signalForSelector 这个方法的作用就是 去监听_blueView的哪个方法的调用,它返回给我们的就是一个信号,直接订阅。这里给我们返回的 x 是一个集合,这个集合里面的数据 就是我们btnClick方法传的参数了。
2.代替KVO
首先在这里我们要监听_blueView的frame属性值的变化,我们需要导入这个头文件
#import <NSObject+RACKVOWrapper.h>
[_blueView rac_observeKeyPath:@"frame" options:NSKeyValueObservingOptionOld |NSKeyValueObservingOptionNew observer:nil block:^(id value, NSDictionary *change, BOOL causedByDealloc, BOOL affectedOnlyLastComponent) {
//回调
NSLog(@"value%@---%@",value,change);
}];
在这里我们直接注册监听,通过Block 就直接回调了我们属性发生变化的事件。这样处理起来就使得我们代码逻辑非常清晰了。
另一种写法
[[_blueView rac_valuesForKeyPath:@"frame" observer:nil] subscribeNext:^(id _Nullable x) {
NSLog(@"%@",x);
}];
3.监听事件
那么在这里 我们在来看看通过RAC如何监听我们的事件呢?
在我们的控制器中有一个按钮“_btn”,那么我们要监听按钮的点击事件。
之前我们一般是怎么做呢?是不是通过_btn的 addTarget:(nullable id) action:(nonnull SEL) forControlEvents:(UIControlEvents)方法,去添加一个事件,然后再去实现这个方法。
那么!来看看 RAC是实现方式
[[_btn rac_signalForControlEvents:(UIControlEventTouchUpInside)] subscribeNext:^(__kindof UIControl * _Nullable x) {
NSLog(@"%@",x);
}];
对 没错 就是这么简单粗暴。就可以在Block中 直接去处理你的 点击事件了。这就是 函数式编程思想。
rac_signalForControlEvents 这个方法就是说 把左边的按钮的什么事件转成一个信号,然后订阅这个信号。
4.代替通知
在这里 我们以监听一个键盘的通知为例,
先想想我们之前的 通知是怎么写的。
是不是...
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(keyboardWasShow:)
name:UIKeyboardWillShowNotification
object:nil];
然后再去实现 keyboardWasShow 这个方法。
那么来看看 我们通过RAC是如何写的
[[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(NSNotification * _Nullable x) {
NSLog(@"%@",x);
}];
是的 没错 就是这样简单 方便,相信这句话就不用解释了吧!
5.监听文本框
监听文本框的输入,实时拿到文本框输入的值。
根据我们系统的方式是 通过实现文本框的协议,去获取。
那么看看 RAC
[_textField.rac_textSignal subscribeNext:^(NSString * _Nullable x) {
NSLog(@"%@",x);
}];
是的 x 就是你文本框输入的值。
6.代替NSTimer
接下来呢,在补充一个东西
在我们以往使用NSTimer 做定时循环执行的时候,
[NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timerMethod) userInfo:nil repeats:YES];
大家有没有遇到过,如果timerMethod正在执行,而此时如果有UI事件的触发,比如滚动我们的屏幕,我们timerMethod执行将会被暂停执行,一旦UI事件执行完毕,timerMethod又会开始执行。原因是我们的NSTimer的事件是交给Runloop去处理,那么Runloop在执行的时候UI模式具有最高优先权。
那要解决这种问题怎么办呢?大家可能会想到,把他放到子线程中去执行, 开启runloop循环。
NSThread * thread = [[NSThread alloc]initWithBlock:^{
NSTimer * timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timerMethod) userInfo:nil repeats:YES];
//Runloop模式 && 多线程!!
//NSDefaultRunLoopMode 默认模式;
//UITrackingRunLoopMode UI模式:只能被UI事件唤醒!!
//NSRunLoopCommonModes 占位模式:默认&UI模式
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
//开启runloop循环
[[NSRunLoop currentRunLoop] run];
}];
[thread start];
还有一种 通过GCD设置timer
@interface ViewController ()
/** */
@property(nonatomic,strong)dispatch_source_t timer;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
//GCD设置timer
dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, dispatch_get_global_queue(0, 0));
//GCD的事件单位是纳秒
dispatch_source_set_timer(timer, DISPATCH_TIME_NOW, 1.0 * NSEC_PER_SEC, 0);
dispatch_source_set_event_handler(timer, ^{
NSLog(@"-----_%@",[NSThread currentThread]);
});
//启动
dispatch_resume(timer);
_timer = timer;
}
@end
这两种方式呢都可以解决我们的问题。可是这样是不是感觉比较麻烦呢?肯定有更简单点儿的,那就是我们RAC。
@interface ViewController ()
@property(nonatomic,strong)RACDisposable * timerDisposable;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
_timerDisposable = [[RACSignal interval:1.0 onScheduler:[RACScheduler scheduler]] subscribeNext:^(NSDate * _Nullable x) {
NSLog(@"%@",[NSThread currentThread]);
}];
}
-(void)dealloc{
//取消订阅!!
[_timerDisposable dispose];
}
@end