其它技术点iOS第三方资料收集MVVM. RAC

ReactiveCocoa-上手其实很简单(一)

2017-03-15  本文已影响326人  Miaoz0070
萌萌哒.jpg

之前写了篇RAC的方法使用,有些小伙伴说还是不会用,太抽象了。网上也有人说RAC的入门门槛高,其实我想说不要被大家误导,入门还是很容易的,下边我们先预热一下来个栗子尝尝:

一、例子:

1.创建信号 :

// 1.创建信号  
    RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {  
        // block调用时刻:每当有订阅者订阅信号,就会调用block。  
        // 发送信号  
        [subscriber sendNext:@"ws"];  
        // 如果不在发送数据,最好发送信号完成,内部会自动调用[RACDisposable disposable]取消订阅信号。  
        [subscriber sendCompleted];  
          
 
        // 取消信号,如果信号想要被取消,就必须返回一个RACDisposable  
        // 信号什么时候被取消:1.自动取消,当一个信号的订阅者被销毁的时候机会自动取消订阅,2.手动取消,  
        //block什么时候调用:一旦一个信号被取消订阅就会调用  
        //block作用:当信号被取消时用于清空一些资源  
        return [RACDisposable disposableWithBlock:^{  
            // block调用时刻:当信号发送完成或者发送错误,就会自动执行这个block,取消订阅信号。  
            // 执行完Block后,当前信号就不在被订阅了。  
            NSLog(@"取消订阅");  
        }];  
    }];  

2.订阅信号

      
    // 2. 订阅信号  
    //subscribeNext  
    // 把nextBlock保存到订阅者里面  
    // 只要订阅信号就会返回一个取消订阅信号的类  
    RACDisposable *disposable = [signal subscribeNext:^(id x) {  
          
        // block调用时刻:每当有信号发出数据,就会调用block.  
         NSLog(@"接收到数据:%@",x);  
    }];  
    // 取消订阅  
    [disposable dispose];  
      

这样就形成RAC一个简单的用法,sender发射"data",将会被receiver的subscribeNext的接收,通过这个例子,也许你会想到“异步”、“观察者模式”,没错,这些都是RAC所做的事情,并且让他们变得更简单和简洁,而RAC所有的一切都将围绕这两个点展开,一个是信号源,一个是订阅者,是不是很通俗易懂?如果你理解了这点或者你已经知道RAC就是这么一回事,那么恭喜你,你已经一只脚跨进RAC的大门了,如果不是!!!!那也无所谓,请继续往下看...

二、ReactiveCocoa作用,也就是说在什么场景使用

在我们iOS开发过程中,经常会响应某些事件来处理某些业务逻辑,例如按钮的点击,上下拉刷新,网络请求,属性的变化(KVO)或者用户位置的变化(通过CoreLocation)。但是这些事件都用不同的方式来处理,比如action、delegate、KVO、callback等。

上边的事件,都可以通过RAC处理,RAC为事件提供了很多处理方法,而且利用RAC处理事件很方便,可以把要处理的事情,和监听的事情的代码放在一起,这样非常方便我们管理,就不需要跳到对应的方法里。使用RAC就不需要考虑调用顺序,直接考虑结果,把每一次操作都写成一系列嵌套的方法中,非常符合我们开发中高聚合,低耦合的思想。

三、各个编程思想对比下之间的差异做个比较:

3.1 面向过程:处理事情以过程为核心,一步一步的实现。

3.2 面向对象:万物皆对象

3.3 链式编程思想:是将多个操作(多行代码)通过点号(.)链接在一起成为一句代码,使代码可读性好。a(1).b(2).c(3)

3.4 响应式编程思想:不需要考虑调用顺序,只需要知道考虑结果,类似于蝴蝶效应,产生一个事件,会影响很多东西,这些事件像流一样的传播出去,然后影响结果,借用面向对象的一句话,万物皆是流。

3.5 函数式编程思想:是把操作尽量写成一系列嵌套的函数或者方法调用。

ReactiveCocoa结合了两种编程风格:

函数式编程(Functional Programming)

响应式编程(Reactive Programming)

所以,RAC被描述为函数响应式编程(FRP)框架。

四、RAC开发中常见用法
 
 代理:
 rac_signalForSelector:用于替代代理。
 
 KVO :
 rac_valuesAndChangesForKeyPath:用于监听某个对象的属性改变。
 
 监听事件:
 rac_signalForControlEvents:用于监听某个事件。
 
 代替通知:
 rac_addObserverForName:用于监听某个通知。
 
 监听文本框文字改变:
 rac_textSignal:只要文本框发出改变就会发出这个信号。
 
 处理当界面有多次请求时,需要都获取到数据时,才能展示界面
 rac_liftSelector:withSignalsFromArray:Signals:当传入的Signals(信号数组),每一个signal都至少sendNext过一次,就会去触发第一个selector参数的方法。
 

想要查看其他方法及修饰符请到<RAC部分方法和修饰符>.

具体代码:
#pragma mark --textField属性变化  
 /*****************创建textField*******************/  
-(void)textFieldChange  
{  
    //初始化一个textField控件  
     UITextField *textField = [[UITextField alloc] init];  
     textField.backgroundColor = [UIColor yellowColor];  
    textField.delegate = self ;  
    [self.view addSubview:textField];  
    @weakify(self);  
    [textField mas_makeConstraints:^(MASConstraintMaker *make) {  
        @strongify(self);  
        make.size.mas_equalTo(CGSizeMake(180, 40));  
        make.center.equalTo(self.view);  
    }];  

    /*****************监听textField的属性变化情况*******************/  
    //RAC内部封装好的类  
    //默认执行1次,所以会有打印,原因:请查看RAC底层bind,hook的思想  
    //监听文本框的文字改变 ,只要textField变化就会得到变化的值。
    [textField.rac_textSignal subscribeNext:^(id x) {  
        //获取到textfield的值
    }];  
  
}  

#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) {  
        //点击调用
        RACViewController *VC = [[RACViewController alloc] init];  
        @strongify(self);  
        [self presentViewController:VC animated:YES completion:^{  
        }];  
    }];  
}  
  
#pragma mark --手势  
-(void)addTap  
{  
    /**************手势*****************/  
    self.view.userInteractionEnabled = YES ;  
    UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc] init];  
    [[tap rac_gestureSignal] subscribeNext:^(id x) {  
     //手势触发调用
    }];  
    [self.view addGestureRecognizer:tap];  
}  
  
#pragma mark --通知中心  
-(void)notifiCenter  
{  
    /**************通知*****************/  
    [[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationDidEnterBackgroundNotification object:nil] subscribeNext:^(id x) {  
      //收到post过来的通知的时候会调用  
    }];  
}  
  
#pragma mark --代理  
-(void)delegateDemo  
{  
    UIAlertView *alterView = [[UIAlertView alloc] initWithTitle:@"RAC" message:@"ReactiveCocoa" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Ensure", nil nil];  
      
    //方法一:代理  
    //    [[self rac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)] subscribeNext:^(id x) {  
    //        RACTuple *tuple = (RACTuple *)x ;  //类似于swift的元组  
    //  
    //        NSLog(tuple);  
    //  
    //        NSLog(tuple.first);  
    //        NSLog(tuple.second);  
    //        NSLog(tuple.third);  
    //  
    //    }];  
      
    [alterView show];  
      
    //方法二:内部封装的方法  
    //更简单的方式  
    [[alterView rac_buttonClickedSignal] subscribeNext:^(id 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) {  
          
        NSLog(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:^{  
    //  
    //        NSLog(rac);  
    //  
    //    }];  
      
    //2.每间隔多长时间做一件  
    [[RACSignal interval:1 onScheduler:[RACScheduler mainThreadScheduler]] subscribeNext:^(id x) {  
          
                NSLog(x);  
        //        NS(@"55555555");  
    }];  
      
      
    //这是定时器最常用的两种方法,第一种方法,延迟时间去做某件事,更改afterDelay的属性,第二种方法,每间隔躲藏时间去做一件事,更改interval的属性  
  
}  
最后

今天就写到这里,相信大家也已经入门了,现在RAC的干货已经很多了,写这些除了帮助想开始学RxJava的新手入门,最重要的还是起到一个笔记的作用吧,好记性不如烂笔头。相互交流。

上一篇 下一篇

猜你喜欢

热点阅读