ReactiveObjC筆記總結

這個方法是說明的例子 下邊會用到很多次

//模擬請求網絡的操作
-(void)loadNetWorkData:(void(^)(id data))success{
    success(@"成功");
}

1:RACMulticastConnection

解決問題:避免多次訂閱一個信號 執行多次信號block內部的代碼
項目中的實戰:對一個網絡請求的信號多次訂閱造成多次請求

//避免因為訂閱多次 導致 信號執行多次
-(void)multicastConnection{
    
    RACSignal *signal = [RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber>  _Nonnull subscriber) {
        [self loadNetWorkData:^(id data) {
            NSLog(@"測試執行次數 --- %@",data);
            [subscriber sendNext:data];
        }];
        return nil;
    }];
    
    RACMulticastConnection *connection = [signal publish];

    [connection.signal subscribeNext:^(id  _Nullable x) {
        NSLog(@"訂閱1 --- %@",x);
    }];

    [connection.signal subscribeNext:^(id  _Nullable x) {
        NSLog(@"訂閱2 ---  %@",x);
    }];

    [connection.signal subscribeNext:^(id  _Nullable x) {
        NSLog(@"訂閱3 --- %@",x);
    }];

    [connection connect];
    
}

2:RACCommand

解決問題: 可以獲取到信號的執行過程.
項目實戰:獲取網絡數據 直接在ViewModel 中用RACCommand封裝,然后在ViewController中拿到執行的狀態 對請求狀態進行操作
坑:
1:訂閱信號 executionSignals 要在 execute 方法之前
2:發送完信號 要發送 sendCompleted 不然command.executing無法接收到信號停止

-(void)commandRAC{
    
    RACCommand *commond = [[RACCommand alloc] initWithSignalBlock:^RACSignal * _Nonnull(id  _Nullable input) {
        NSLog(@"1:開始執行input ---- %@",input);
        return [RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber>  _Nonnull subscriber) {
            [subscriber sendNext:@"345"];
            [subscriber sendCompleted];
            return nil;
        }];
    }];
    
    [commond.executionSignals.switchToLatest subscribeNext:^(id  _Nullable x) {
        NSLog(@"3:拿到最新的信號 --- %@",x);
    }];
    
    [commond execute:@"執行的傳值"];

    //跳過第一次  [commond.executing skip:1];
    [[commond.executing skip:1] subscribeNext:^(NSNumber * _Nullable x) {
        BOOL isExecuting = [x boolValue];
        if (isExecuting) {
            NSLog(@"2:commond 正在執行");
        }else{
            NSLog(@"4:commond 執行結束");
        }
    }];
    
}

3: rac_liftSelector

功能: 可以檢測到幾個信號全部執行完載執行接下來的方法
項目實戰:解決UI展示需要多個接口執行完才能繼續執行接下來的方法!(還可以通過GCD的線程組完成 不過這個更直觀點)

坑:
[self rac_liftSelector:@selector(reloadComplationDataA:dataB:) withSignalsFromArray:@[signalA,signalB]];
載執行 上邊 @selector()方法的時候有幾個信號就需要幾個參數
-(void)waitingLoadAllData{
    
    @weakify(self);
    RACSignal *signalA = [RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber>  _Nonnull subscriber) {
        @strongify(self);
        //延遲三秒做操作
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            [self loadNetWorkData:^(id data) {
                NSLog(@"第一個網絡請求結束--%@",data);
                [subscriber sendNext:@"data"];
            }];
        });
        return nil;
    }];
    
    RACSignal *signalB = [RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber>  _Nonnull subscriber) {
        @strongify(self);
        [self loadNetWorkData:^(id data) {
            NSLog(@"第二個網絡請求結束--%@",data);
            [subscriber sendNext:@"data"];
        }];
        return nil;
    }];
    
    [self rac_liftSelector:@selector(reloadComplationDataA:dataB:) withSignalsFromArray:@[signalA,signalB]];
    
}

-(void)reloadComplationDataA:(id)dataA dataB:(id)dataB{
    NSLog(@"全部請求完成");
}

2019-05-07 16:04:00.107081+0800  第二個網絡請求結束--成功
2019-05-07 16:04:03.107146+0800  第一個網絡請求結束--成功
2019-05-07 16:04:03.107364+0800  全部請求完成

4:信號之間的依賴 then

功能: 執行完本次信號的操作再去執行下一個操作
項目實戰:執行完本次網絡操作 才能接著執行下次網絡操作,類似京東的分類頁面。需要父分類的 id 才去請求父分類下子分類的數據
避免了 block 的循環嵌套操作


-(void)replyonOtherSignal{
    [[[self loadCatagory] then:^RACSignal * _Nonnull{
        return [self loadDetail];
    }] subscribeNext:^(id  _Nullable x) {
        NSLog(@"完成 ---- %@",x);
    }];
}

-(RACSignal *)loadCatagory{
    RACSubject *signa = [RACReplaySubject subject];
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        [self loadCatagoryData:^(id data) {
            NSLog(@"加載分類完成 ---- %@",data);
            [signa sendNext:data];
            [signa sendCompleted];
        }];
    });
    return signa;
}

-(RACSignal *)loadDetail{
    RACSubject *signa = [RACReplaySubject subject];
    [self loadCatagoryDetail:^(id data) {
        //模擬延遲加載
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            NSLog(@"加載詳情完成 ---- %@",data);
            [signa sendNext:data];
        });
    }];
    return signa;
}

-(void)loadCatagoryData:(void(^)(id data))success{
    success(@"123");
}

-(void)loadCatagoryDetail:(void(^)(id data))success{
    success(@"456");
}

5:merge

可以調整兩個信號的執行順序 block 執行兩邊

-(void)merge{
    RACSubject *signalA = [RACSubject subject];
    RACSubject *signalB = [RACSubject subject];
    
    [[signalA merge:signalB] subscribeNext:^(id  _Nullable x) {
        NSLog(@"%@",x);
    }];
    
    [signalA sendNext:@"A"];
    [signalB sendNext:@"B"];
}

6:zip


-(void)zip{
    RACSubject *signalA = [RACSubject subject];
    RACSubject *signalB = [RACSubject subject];
    
    [[signalA zipWith:signalB] subscribeNext:^(id  _Nullable x) {
        //把數據合并到一起 同事傳輸數據  數據是元祖的方式
        //發送信號數量 必須大于1
        RACTupleUnpack(NSString *A , NSString *B) = x;
        NSLog(@"x === %@ A === %@ B === %@",x,A,B);
    }];
    
    [signalA sendNext:@"A"];
    [signalB sendNext:@"B"];
}

7:Bind

loginButton 登錄按鈕
userNameTF 用戶名的輸入框
passWordTF 密碼的輸入框
綁定按鈕的 點擊狀態 (如果 userNameTF passWordTF 有值loginButton.enable = YES (同理 NO) )

    RAC(self.loginButton,enabled) = [RACSignal combineLatest:@[self.userNameTF.rac_textSignal,self.passWordTF.rac_textSignal] reduce:^id(NSString *userName,NSString *password){
        return @(userName.length && password.length);
    }];

8:replay

作用:重復執行
項目實戰:如果 token 請求失敗 需要接著請求幾次,但是不能一直請求 會設置最大請求次數 這個就能解決

-(void)replay{
    
    __block NSInteger index = 1;
    [[[RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber>  _Nonnull subscriber) {
        if (index == 5) {
            [subscriber sendNext:@"重復"];
        }else{
            NSError *error = [NSError errorWithDomain:NSDocumentTypeDocumentOption code:-8080 userInfo:@{@"error":@"錯誤"}];
            [subscriber sendError:error];
        }
        index++;
        return nil;
    }] retry] subscribeNext:^(id  _Nullable x) {
        NSLog(@"成功 --- %@",x);
    }];
}

9:filter

過濾操作


-(void)filter{
    [[self.passWord.rac_textSignal filter:^BOOL(NSString * _Nullable value) {
        return value.length > 6;
    }] subscribeNext:^(NSString * _Nullable x) {
        NSLog(@"%@",x);
    }];
}
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容