MVVM

RAC基本用法

2017-04-19  本文已影响52人  天外丶飞仙

最近公司要用MVVM+RAC重构项目,所以就对RAC开发中的基本用法总结了一下。

RACSignal

RACSignal使用步骤: 1.创建信号 2.订阅信号 3.发送信号。

RACSignal * signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    //信号被订阅就会执行这个block
    // subscribe调用:只要一个信号被订阅就会调用
    // subscribe作用:发送数据       
    [subscriber sendNext:@"发送数据"];
    //发送完成
    [subscriber sendCompleted];
    //返回值是RACDisposable这个类
    return [RACDisposable disposableWithBlock:^{
        // block调用时刻:当信号发送完成或者发送错误,就会自动执行这个block,取消订阅信号。
        // 执行完Block后,当前信号就不在被订阅了。
        NSLog(@"信号被销毁");
    }];
}];
[signal subscribeNext:^(id x) {
    // nextBlock调用:只要订阅者发送数据就会调用
    // nextBlock作用:处理数据,展示到UI上面
    // x:信号发送的内容
    NSLog(@"%@",x);
}];

RACSubject

RACSubject使用步骤: 1.创建信号 2.订阅信号 3.发送信号。

// 1.创建信号
RACSubject *subject = [RACSubject subject] ;
// 2.订阅信号
[subject subscribeNext:^(id x) {
    // block调用时刻:当信号发出新值,就会调用.
    NSLog(@" 接受到的信号%@",x);
}];
// 3.发送信号
[subject sendNext:@"1"];

RACReplaySubject

RACReplaySubject使用步骤和RACSubject相同,区别在于订阅信号,发送信号可以交换顺序。

// 1.创建信号
RACReplaySubject *replaySubject = [RACReplaySubject subject];
// 2.发送信号
[replaySubject sendNext:@1];
// 3.订阅信号
[replaySubject subscribeNext:^(id x) {
    NSLog(@"订阅者接收到的数据%@",x);
}];

RACSequence

可以用RACSequence来遍历数组。

NSArray *numbers = @[@1,@2,@3,@4];
[numbers.rac_sequence.signal subscribeNext:^(id x) {
       //拿到遍历的数组值
       NSLog(@"%@",x);
}];

用来遍历字典

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

RACMulticastConnection

假设在一个信号中发送请求,每次订阅一次都会发送请求,这样就会导致多次请求。使用RACMulticastConnection不管订阅多少次,请求只发送一次。

  // 1.创建信号
RACSignal *signal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    NSLog(@"发送请求");
    [subscriber sendNext:@1];
    return nil;
}];
// 2.创建连接
RACMulticastConnection *connect = [signal publish];
// 3.订阅信号,
[connect.signal subscribeNext:^(id x) {
    NSLog(@"订阅者一");
}];
[connect.signal subscribeNext:^(id x) {
    NSLog(@"订阅者二");
}];
// 4.连接,激活信号
[connect connect];

RAC开发中常用的用法

代替代理:rac_signalForSeletor:
代替KVO:rac_valuesAndChangesForKeyPath:
监听事件:rac_signalForControlEvents:
代替通知: rac_addObbserverForName:
监听文本框文字改变: rac_textSignal:
当一个界面多次请求时,需要都请求完成才搭建界面:
//方法的参数必须和数组的参数一一对应
ran_liftSelector: withSignalsFromArray:

RAC中常用的宏

//1. 用来给某个对象的某个属性绑定信号,只要产生信号内容,就会把内容给属性赋值
RAC(_label,text) = _textField.rac_textSignal;

//2.只要这个对象的属性一改变就会产生信号
[RACObserve(self.view, frame) subscribeNext:^(id x) {
    NSLog(@"%@",x);
}];

//3. 包装元组
RACTuple *tuple = RACTuplePack(@1,@2);

//4.@weakify() @strongify()用来解决循环引用

RACCommand

在开发中常用于处理事件,比如网络请求。

// 1.创建命令
RACCommand *command = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) {
    // input:执行命令传入参数
    // Block调用:执行命令的时候就会调用
    NSLog(@"%@",input);
    return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
     // 发送数据
      [subscriber sendNext:@"执行命令产生的数据"];
        return nil;
    }];
}];
// 2.执行命令
RACSignal *signal = [command execute:@1];
// 3.订阅信号拿到信号中发送的数据。
[signal subscribeNext:^(id x) {
    NSLog(@"%@",x);
}];
//这个方法也能拿到信号中发送的数据
[command.executionSignals.switchToLatest subscribeNext:^(id x) {
    NSLog(@"%@",x);
}];
// 监听事件有没有完成
[command.executing subscribeNext:^(id x) {
    if ([x boolValue] == YES) { // 当前正在执行
        NSLog(@"当前正在执行");
    }else{
        // 执行完成/没有执行
        NSLog(@"执行完成/没有执行");
    }
}];

RAC中的映射

flattenMap用于信号中的信号

 // 创建信号
RACSubject *subject = [RACSubject subject];
// 绑定信号
RACSignal *bindSignal = [subject flattenMap:^RACStream *(id value) {
    // block:只要源信号发送内容就会调用
    // value:就是源信号发送内容
    value = [NSString stringWithFormat:@"xmg:%@",value];
    // 返回信号用来包装成修改内容值
    return [RACReturnSignal return:value];
}];
// flattenMap中返回的是什么信号,订阅的就是什么信号
// 订阅信号
[bindSignal subscribeNext:^(id x) {
    NSLog(@"%@",x);
}];
// 发送数据
[subject sendNext:@"123"];

map

 // 创建信号
RACSubject *subject = [RACSubject subject];
// 绑定信号
RACSignal *bindSignal = [subject map:^id(id value) {
    // 返回的类型,就是你需要映射的值
    return [NSString stringWithFormat:@"haha:%@",value];
}];
// 订阅绑定信号
[bindSignal subscribeNext:^(id x) {
    NSLog(@"%@",x);
}];
[subject sendNext:@"1"];

RAC信号组合

concat

// 创建信号A
RACSignal *siganlA = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    // 发送请求
    NSLog(@"发送上部分请求");
    // 发送信号
    [subscriber sendNext:@"上部分数据"];
    [subscriber sendCompleted];
    return nil;
}];
RACSignal *siganlB = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    // 发送请求
    NSLog(@"发送下部分请求");
    // 发送信号
    [subscriber sendNext:@"下部分数据"];
    return nil;
}];
// concat:按顺序去连接
// 注意:concat,第一个信号必须要调用sendCompleted
// 创建组合信号
RACSignal *concatSignal = [siganlA concat:siganlB];
// 订阅组合信号
[concatSignal subscribeNext:^(id x) {
    // 既能拿到A信号的值,又能拿到B信号的值
    NSLog(@"%@",x);
}];

combineLatest

// 组合    
// reduceBlock参数:根组合的信号有关,一一对应
RACSignal *comineSiganl = [RACSignal combineLatest:@[_accountFiled.rac_textSignal,_pwdField.rac_textSignal] reduce:^id(NSString *account,NSString *pwd){
    // block:只要源信号发送内容就会调用,组合成新一个值
    NSLog(@"%@ %@",account,pwd);
    // 聚合的值就是组合信号的内容
    return @(account.length && pwd.length);
}];
 // 订阅组合信号
RAC(_loginBtn,enabled) = comineSiganl;

RAC过滤

filte过滤信号

// 只有当我们文本框的内容长度大于5,才想要获取文本框的内容
[[_textField.rac_textSignal filter:^BOOL(id value) {
    // value:源信号的内容
    return  [value length] > 5;
    // 返回值,就是过滤条件,只有满足这个条件,才能能获取到内容
}] subscribeNext:^(id x) {
    NSLog(@"%@",x);
}];

ignore:忽略一些值

 // 1.创建信号
RACSubject *subject = [RACSubject subject];
// 2.忽略一些
RACSignal *ignoreSignal = [subject ignoreValues];
// 3.订阅信号
[ignoreSignal subscribeNext:^(id x) {
    NSLog(@"%@",x);
}];
// 4.发送数据
[subject sendNext:@"13"];

skip;跳跃几个值

 // 1.创建信号
RACSubject *subject = [RACSubject subject];
[[subject skip:2] subscribeNext:^(id x) { 
    NSLog(@"%@",x);
}];
[subject sendNext:@"1"];
[subject sendNext:@"2"];
[subject sendNext:@"3"];
上一篇 下一篇

猜你喜欢

热点阅读