RAC自己练习下

ReactiveCocoa使用个人总结

ReactiveCocoa简介

ReactiveCocoa(简称RAC)是由Github开源的一个应用于iOS和OS开发的新框架。

ReactiveCocoa的作用

在我们iOS开发过程中,当某些事件响应的时候,需要处理某些业务逻辑,这些事件都会用不同的方式来处理。比如:按钮的点击使用action、UIScrollView的滚动使用的delegate、监测属性的值改变使用KVO等系统提供的方式。

RAC就是把这些不同的响应事件的方式统一起来,使这些方式都可以通过RAC来处理。

RAC为事件提供了很多的处理方法,而且利用RAC来处理事件很方便,可以把要处理的事情和监听的代码放一起,方便我们的管理,这样就不需要跳转到对应的方法里去处理,非常符合我们开发中高内聚,低耦合的思想。

RAC在某些特定情况下开发时可以大大简化代码,提高开发效率,并且是安全可靠的。

配置RAC环境

使用cocoapods来获取github上的开源库:

pod 'ReactiveCocoa'

如何使用RAC

  • RACSignal

信号类,一般表示将来会有数据传递,只要数据改变,信号内部接收到数据,就会马上发出数据。

  • RACSignal只是表示当数据改变时,信号内部会发出数据,他本身不具备发送信号的能力,而是交给内部的一个订阅者subscriber发出。
  • 默认一个信号都是冷信号,也就是说值改变了也不会触发,只有订阅了这个信号,值改变了才会触发。
  • 订阅信号:调用RACSignal的subscribeNext:方法就可以订阅信号。
    // 创建一个RACSignal信号
    RACSignal * signal = [RACSignal createSignal:^RACDisposable *(id subscriber) {
                  // 每当有订阅者订阅信号  就会调用这个block
                  // 发送信号
                  [subscriber sendNext:@"Hello"];
                  // 如果不再发送数据,最好调用此方法发送信号完成
                  [subscriber sendCompleted];
                  return [RACDisposable disposableWithBlock:^{
                        // 当信号发送完成或者发送错误,就会自动执行这个block,取消订阅信号
                        NSLog(@"信号被销毁");
            }];
    }];
    
    // 订阅信号,才会激活信号
    [signal subscribeNext:^(id x){
        // 每当有信号发出数据  就会调用block
        NSLog(@"接收到数据%@",x);
    }]

  • target-action

RAC最基本的使用就是对事件的监听。我们来看以下代码:

RACSignal * textFieldSignal = [textField rac_signalForControlEvents:UIControlEventEditingChanged];
[textFieldSignal subscribeNext:^(id x) {
        NSLog(@"变化的text:%@",x);
}];

这段代码实现了监听UITextFieldUIControlEventEditingChanged事件,当textField里的内容发生变化时,发送变化的数据调用block种的内容将变化的内容实时打印出来。这两段代码通常会连在一起写:

[[textField rac_signalForControlEvents:UIControlEventEditingChanged] subscribeNext:^(id x) {
        NSLog(@"变化的text:%@",x);
    }];

同样的,我们以这段代码为模板进行RAC的使用,UIButton的点击事件我们也可以用RAC进行监测,添加方法,对于UITextField的文字更改的监听也有更简单的写法:

// UIButton添加RAC监听点击方法
[[button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
        NSLog(@"点击button:%@",x);
}];
// 对于UITextField 文字更改的监听
[textField.rac_textSignal subscribeNext:^(id x) {
        NSLog(@"变化的text:%@",x);
}]

RAC的一些常见操作方法

  1. ReactiveCocoa操作方法之映射(flattenMap,Map)

    • flattenMap,Map用于把源信号内容映射成新的内容。

    flattenMap的使用:把源信号的内容映射成一个新的信号,信号可以是任意类型 flattenMap中的block返回的是信号

```
// 创建信号中的信号

RACSubject * signalOfSignals = [RACSubject subject];
RACSubject * signals = [RACSubject subject];

[[signalOfSignals flattenMap:^RACStream *(id value) {
    // 当signalOfsignals 发出signals 信号才会调用  这里的value返回的是信号
    return value;
}] subscribeNext:^(id x) {
    // 只有signalOfSignals发出的signals也发出了内容(也就是最后一行代码执行)才会调用
    NSLog(@"=========>%@",x);
}];

// 信号中的信号发送一个信号
[signalOfSignals sendNext:signals];
// 信号发送一个内容
[signals sendNext:@"信号发送信号"];

```

> map的使用:把源信号的值映射成一个新的值,`map`中的block返回的是一般对象(这里一般用于把源信号里的json数据映射成相应的model)

```
// 监听文本框内容的改变,映射成一个新的值
[[textField.rac_textSignal map:^id(id value) {
    // 这里是把文本框的内容加上 “Hello RAC” 的前缀
    return [NSString stringWithFormat:@"Hello RAC :%@",value];
}] subscribeNext:^(id x) {
    NSLog(@"%@",x);
}];
```

> 开发中,如果信号发出的不是信号,映射一般使用`map`,如果信号发出的是信号,则使用`flattenMap`。
  1. ReactiveCocoa操作方法之组合

    • concat:按一定顺序拼接信号,当多个信号发出的时候,有顺序的接收信号
     RACSignal * signalA = [RACSignal createSignal:^RACDisposable *(id subscriber) {
        [subscriber sendNext:@"First"];
        [subscriber sendCompleted];
        return nil;
    }];
    
    RACSignal * signalB = [RACSignal createSignal:^RACDisposable *(id subscriber) {
        [subscriber sendNext:@"Second"];
        [subscriber sendCompleted];
        return nil;
    }];
    
    // 把A拼接到B后   A发送完成B才会被激活  按顺序
    RACSignal * concatSignal = [signalA concat:signalB];
    [concatSignal subscribeNext:^(id x) {
        NSLog(@"Concat Signal %@",x);
    }];
    
    • then:用于连接两个信号,当第一个信号完成,才会连接then返回的信号,第一个信号会被忽略掉。
     RACSignal * thenSignal = [signalA then:^RACSignal *{
        return signalB;
    }];
    
    [thenSignal subscribeNext:^(id x) {
        // 此处输出为 Then signal Second   第一个信号会被忽略掉
        NSLog(@"Then signal %@",x);
    }];
    
    • merge把多个信号合并为一个信号,任何一个信号有新值就会调用。
    RACSignal * mergeSignal = [signalA merge:signalB];
    [mergeSignal subscribeNext:^(id x) {
        NSLog(@"Merge Signal %@",x);
    }];
    
    • zipWith:用于把两个信号压缩成一个信号,只有两个信号都发出信号内容时,把两个信号的内容合并成一个元组,才会触发压缩流的next事件。
    RACSignal * zipWithSignal = [signalA zipWith:signalB];
    [zipWithSignal subscribeNext:^(id x) {
        NSLog(@"ZipWith Signal %@",x);
    }];
    
    • combineLatest:将多个信号合并起来,并且拿到各个信号的最新的值,必须每个合并的signal至少都有过一次sendNext,才会触发合并的信号。这个方法试了一下,觉得跟zipWith是相似的。
    RACSignal * combineSignal = [signalA combineLatestWith:signalB];
    [combineSignal subscribeNext:^(id x) {
        NSLog(@"Combine Signal %@",x);
    }];
    
    • reduce:聚合:用于信号发出的内容是元组,把信号发出元组的值聚合成一个值
    // 常见的方法(先组合再聚合)。
    // 先组合再聚合,把上面两个信号聚合在一起。形成一个信号  (两者都成立才为yes)
    // 用于登录页面,两个文本框的内容都符合条件,登录按钮才会响应
        RACSignal * reduceSignal = [RACSignal combineLatest:@[usernameLengthSignal, passwordLengthSignal] reduce:^id(NSNumber * username , NSNumber * password){
            return @(username.boolValue && password.boolValue);
        }];
        
    

RAC一些常用的宏

  1. RAC(),这个宏的完整写法是:RAC(TARGET,[KEYPATH,NIL_VALUE])

    使用方法:

     RAC(label,text) = self.textField.rac_textSignal;
     RAC(label,text,@"如果是nil就显示这里的文字") = self.textField.rac_textSignal;
    

    最常用的宏RAC()一般出现在等号左边,右边是一个RACSignal,作用是将一个对象(这里是label)的一个属性(这里是label的text)和一个RACSignal绑定,RACSignal每产生一个value,都会自动将这个对象的这个属性设置为value。如果value是nil,就设置为第三个参数。(上面的代码的效果是把label的text和textField的文字变动的signal绑定, 每当textField里面的文字有变化,就给label设置新的text。)

  2. RACObserve(TARGET,KEYPATH)

    作用是观察TARGET的KEYPATH属性,产生一个RACSignal,相当于KVO。

    RAC(self.outputLabel, text) = RACObserve(self.model, name); 
    

    把label的text和model的name绑定,只要name有变化,label的text也会变化为相应的值。

暂时写这么点吧。 代码点击这里.

你可能感兴趣的:(RAC自己练习下)