這個方法是說明的例子 下邊會用到很多次
//模擬請求網絡的操作
-(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);
}];
}