ReactiveCocoa方法总结

作者: 林希品 | 来源:发表于2022-01-07 09:07 被阅读0次

    基础部分

    /*************************************基础部分**************************************************/
    #pragma mark --textField属性变化
    -(void)textFieldChange
    {
        //初始化一个textField控件
        UITextField *textField = ({
            UITextField *textField = [[UITextField alloc] init];
            textField.backgroundColor = [UIColor cyanColor];
            textField ;
        });
        textField.delegate = self ;
        [self.view addSubview:textField];
        [textField mas_makeConstraints:^(MASConstraintMaker *make) {
            make.size.mas_equalTo(CGSizeMake(180, 40));
            make.center.equalTo(self.view);
        }];
        
        
        /*****************监听textField的属性变化情况*******************/
        //RAC内部封装好的类
        //默认执行1次,所以会有打印,原因:请查看RAC底层bind,hook的思想
        //监听文本框的文字改变
        [textField.rac_textSignal subscribeNext:^(id x) {
            NSLog(x);
        }];
    }
     
     
    #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) {
            LxPrintf(@"进来了");
            RACViewController *VC = [[RACViewController alloc] init];
            @strongify(self);
            [self presentViewController:VC animated:YES completion:^{
                LxPrintf(@"%s",__func__);
            }];
        }];
    }
     
     
    #pragma mark --手势
    -(void)addTap
    {
        /**************手势*****************/
        self.view.userInteractionEnabled = YES ;
        UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc] init];
        [[tap rac_gestureSignal] subscribeNext:^(id x) {
            LxDBAnyVar(x);
        }];
        [self.view addGestureRecognizer:tap];
    }
     
    #pragma mark --通知中心
    -(void)notifiCenter
    {
        /**************通知*****************/
        [[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationDidEnterBackgroundNotification object:nil] subscribeNext:^(id x) {
            LxDBAnyVar(x);
        }];
    }
     
     
     
    #pragma mark --代理
    -(void)delegateDemo
    {
        UIAlertView *alterView = [[UIAlertView alloc] initWithTitle:@"RAC" message:@"ReactiveCocoa" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Ensure", nil];
        
        //方法一:代理
        //    [[self rac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)] subscribeNext:^(id x) {
        //
        //        RACTuple *tuple = (RACTuple *)x ;  //类似于swift的元组
        //        LxDBAnyVar(tuple);
        //        LxDBAnyVar(tuple.first);
        //        LxDBAnyVar(tuple.second);
        //        LxDBAnyVar(tuple.third);
        //    }];
        [alterView show];
        
        
        //方法二:内部封装的方法
        //更简单的方式
        [[alterView rac_buttonClickedSignal] subscribeNext:^(id x) {
            LxDBAnyVar(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) {
            LxDBAnyVar(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:^{
        //        LxDBAnyVar(rac);
        //    }];
        
        //2.每间隔多长时间做一件
        [[RACSignal interval:1 onScheduler:[RACScheduler mainThreadScheduler]] subscribeNext:^(id x) {
                    LxDBAnyVar(x);
            //        LxPrintAnything(@"55555555");
        }];
        //这是定时器最常用的两种方法,第一种方法,延迟时间去做某件事,更改afterDelay的属性,第二种方法,每间隔躲藏时间去做一件事,更改interval的属性
    }
    

    1、RACSubject当做代理使用

    步骤一:在第二个控制器.h,添加一个RACSubject代替代理。

    @interface TwoViewController : UIViewController
    
    @property (nonatomic, strong) RACSubject *delegateSignal;
    
    @end
    

    步骤二:监听第二个控制器按钮点击

    @implementation TwoViewController
    - (IBAction)notice:(id)sender {
        // 通知第一个控制器,告诉它,按钮被点了
         // 通知代理
         // 判断代理信号是否有值
        if (self.delegateSignal) {
            // 有值,才需要通知
            [self.delegateSignal sendNext:nil];
        }
    }
    @end
    

    步骤三:在第一个控制器中,监听跳转按钮,给第二个控制器的代理信号赋值,并且监听.

    @implementation OneViewController 
    - (IBAction)btnClick:(id)sender {
        // 创建第二个控制器
        TwoViewController *twoVc = [[TwoViewController alloc] init];
        // 设置代理信号
        twoVc.delegateSignal = [RACSubject subject];
        // 订阅代理信号
        [twoVc.delegateSignal subscribeNext:^(id x) {
            NSLog(@"点击了通知按钮");
        }];
        // 跳转到第二个控制器
        [self presentViewController:twoVc animated:YES completion:nil];
    }
    @end
    

    2、RACSubject遍历 map处理

    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);
        }];
    

    2.遍历字典,遍历出来的键值对会包装成RACTuple(元组对象)

        NSDictionary *dict = @{@"name":@"xmg",@"age":@18};
        [dict.rac_sequence.signal subscribeNext:^(RACTuple *x) {
            // 解包元组,会把元组的值,按顺序给参数里面的变量赋值
            RACTupleUnpack(NSString *key,NSString *value) = x;
            // 相当于以下写法
    //        NSString *key = x[0];
    //        NSString *value = x[1];
            NSLog(@"%@ %@",key,value);
        }];
    

    3.字典转模型

    3.1 OC写法

        NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test.plist" ofType:nil];
        NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath];
        NSMutableArray *items = [NSMutableArray array];
        for (NSDictionary *dict in dictArr) {
            TestModel *item = [TestModel yy_modelWithDictionary:dict];
            [items addObject:item];
        }
    

    3.2 RAC写法

        NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test.plist" ofType:nil];
        NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath];
        NSMutableArray *flags = [NSMutableArray array];
        // rac_sequence注意点:调用subscribeNext,并不会马上执行nextBlock,而是会等一会。
        [dictArr.rac_sequence.signal subscribeNext:^(id x) {
            // 运用RAC遍历字典,x:字典
            TestModel *item = [TestModel yy_modelWithDictionary:dict];
            [flags addObject:item];
        }];
    

    3.3 RAC高级写法:

        NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test.plist" ofType:nil];
        NSArray *dictArr = [NSArray arrayWithContentsOfFile:filePath];
        // map:映射的意思,目的:把原始值value映射成一个新值
        // array: 把集合转换成数组
        // 底层实现:当信号被订阅,会遍历集合中的原始值,映射成新值,并且保存到新的数组里。
        NSArray *flags = [[dictArr.rac_sequence map:^id(id value) {
            return [TestModel yy_modelWithDictionary: value];
        }] array];
    

    小白遍历 map学习处理

    小白遍历 map学习处理

    小白遍历 map学习处理

    NSArray *numberTwo = @[@(123),@(123),@(123),@(123)];
    NSLog(@"before = number = %@",numberTwo);
    NSArray *mapArray = [[numberTwo.rac_sequence map:^id(id value) {
      int changeValue = [value intValue];
      return [NSString stringWithFormat:@"%d",changeValue +4];
    }] array];
     NSLog(@"affter = number = %@",mapArray);
    

    其余常见用法

    1 代替KVO :

    rac_valuesAndChangesForKeyPath:用于监听某个对象的属性改变。
    
    // 把监听redV的center属性改变转换成信号,只要值改变就会发送信号
        // observer:可以传入nil
        [[redV rac_valuesAndChangesForKeyPath:@"center" options:NSKeyValueObservingOptionNew observer:nil] subscribeNext:^(id x) {
    
            NSLog(@"%@",x);
    
        }];
    
    

    2 监听事件:

    rac_signalForControlEvents:用于监听某个事件。
    // 把按钮点击事件转换为信号,点击按钮,就会发送信号
    
        [[self.btn rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
    
            NSLog(@"按钮被点击了");
        }];
    

    3 代替通知:

    rac_addObserverForName:用于监听某个通知。
    
    [[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIKeyboardWillShowNotification object:nil] subscribeNext:^(id x) {
            NSLog(@"键盘弹出");
        }];
    
    

    4 监听文本框文字改变:

    rac_textSignal:只要文本框发出改变就会发出这个信号。
    
    [_textField.rac_textSignal subscribeNext:^(id x) {
      NSLog(@"文字改变了%@",x);
    }];
    
    [[self.testTextField.rac_textSignal map: ^id _Nullable(NSString* value) {
      return [NSString stringloithFormat:@"map自定义了返回信号:%0",value ]; //改变文字输出
    }] subscribeNext:^(id _Nullable x) {
      NSLog(@"%@", x);
    }];
    

    5.代理
    第一种:代替代理

    UIAlertView *alterView = [[UIAlertView alloc] initWithTitle:@"RAC" message:@"ReactiveCocoa" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Ensure", nil];
        
      
    //以UIAlertViewDelegate为例用rac_signalForSelector
    //实现- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex代理方法
    [[self rac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)] subscribeNext:^(RACTuple * _Nullable tup) {
            if(tup){
                RACTupleUnpack(UIAlertView *alert,NSNumber *buttonIndex) = tup;
                NSInteger seIndex = [buttonIndex integerValue];
                NSString *btnTitle = [alert buttonTitleAtIndex:seIndex];
                if(alert.tag == 1000){
                     NSLog(@"btnTitle = %@",btnTitle);
                    }
                }
            }
     }];
    ;
        
        [alterView show];
    

    第二种:监听某个对象调用某个方法

    // RAC方法:可以判断某个方法有没有调用
    // 只要self调用Selector就会产生一个信号
    // rac_signalForSelector:监听某个对象调用某个方法
    [[self rac_signalForSelector:@selector(loadAdvertisementsData)] subscribeNext:^(id x) {
      NSLog(@"控制器调用了loadAdvertisementsData");
    }];
    // 判断下_btn有没有调用btnClick,有信号就表示点击了按钮
    [[_btn rac_signalForSelector:@selector(btnClick:)] subscribeNext:^(id x) {
       NSLog(@"点击了按钮");
    }];
    
    

    相关文章

      网友评论

        本文标题:ReactiveCocoa方法总结

        本文链接:https://www.haomeiwen.com/subject/rljrcrtx.html