ReactiveCocoa入门学习二(RAC常见用法)

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 
[_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

你可能感兴趣的:(ReactiveCocoa入门学习二(RAC常见用法))