RAC 常用方法目錄(收集)

RAC

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.
    
    /*******************************第一部分----簡單使用*******************************/
    //文本框事件:
    [self textFiledTest];
    //手勢
    [self gestureTest];
    //通知
    [self notificationTest];
    //定時器NSTime
    [self timeTest];
    //代理 (有局限,只能取代沒有返回值的代理方法)
    [self delegateTest];
    //KVO
    [self kvoTest];
    /*******************************第二部分----進階*******************************/
    //雙向綁定
    [self banding];
    //信號 (創建信號 & 激活信號 & 廢棄信號)
    [self createSignal];
    //信號的處理
    //map (映射)和filter
    [self mapAndFilter];
    //delay延遲
    [self delay];
    //最先開始的時候
    [self startWith];
    //超時
    [self timeOut];
    //take skip
    [self takeOrSkip];
    //throttle(截流)  結合即時搜索優化來講
    [self throttle];
    //repeat 重復
    [self repeatTest];
    //merge 合并信號
    [self mergeTest];
    //RAC(TARGET, ...) 宏
    [self RAC];
    //rac做一個秒表
    [self stopwatch]; 
    //壓縮同合并     ab: ab=>
    [self zipWith];
    //活合并       ab: a=> b=>
    [self merge];
    //過濾合并      ab: b=>
    [self then];
    //順序合并      ab: a->b=>
    [self concat];
}
 #pragma mark 0.雙向綁定
- (void)banding{
    RACChannelTerminal *channelA = RACChannelTo(self, valueA);
    RACChannelTerminal *channelB = RACChannelTo(self, valueB);
    [[channelA map:^id(NSString *value) {
        if ([value isEqualToString:@"西"]) {
                return @"東";
            }
        return value;
    }] subscribe:channelB];
    [[channelB map:^id(NSString *value) {
        if ([value isEqualToString:@"左"]) {
                return @"右";
            }
        return value;
    }] subscribe:channelA];

    [RACObserve(self, valueA) subscribeNext:^(NSString* x) {
        NSLog(@"你向%@", x);
    }];
    [RACObserve(self, valueB) subscribeNext:^(NSString* x) {
        NSLog(@"他向%@", x);
    }];
    self.valueA = @"西";
    self.valueB = @"左";
}

 #pragma mark 1.文本框事件
- (void)textFiledTest{
    
    UITextField * textField = ({
        UITextField * textField = [[UITextField alloc]init];
        textField.backgroundColor = [UIColor cyanColor];
        
        textField;
    });
   [self.view addSubview:textField];
    
    @weakify(self); //  __weak __typeof__(self) self_weak_ = self;
    
    [textField mas_makeConstraints:^(MASConstraintMaker *make) {
        
        @strongify(self);    // __strong __typeof__(self) self = self_weak_;
        make.size.mas_equalTo(CGSizeMake(180, 40));
        make.center.equalTo(self.view);
    }];
    
    [[textField rac_signalForControlEvents:UIControlEventEditingChanged]
     subscribeNext:^(id x) {
         
         LxDBAnyVar(x);
     }];
    //更簡單的方式
    [textField.rac_textSignal subscribeNext:^(NSString *x) {
        
        LxDBAnyVar(x);
    }];
}

#pragma mark 2.手勢
- (void)gestureTest
{
    self.view.userInteractionEnabled = YES;
    UITapGestureRecognizer * tap = [[UITapGestureRecognizer alloc]init];
    [[tap rac_gestureSignal] subscribeNext:^(UITapGestureRecognizer * tap) {
        
        LxDBAnyVar(tap);
    }];
    [self.view addGestureRecognizer:tap];
}

 #pragma mark 3.通知
- (void)notificationTest
{
    [[[NSNotificationCenter defaultCenter] rac_addObserverForName:UIApplicationDidEnterBackgroundNotification object:nil] subscribeNext:^(NSNotification * notification) {
        
        LxDBAnyVar(notification);
    }];
    //不需要removeObserver
}
 
#pragma mark 4.定時器
- (void)timeTest
{
    //1. 延遲某個時間后再做某件事
    [[RACScheduler mainThreadScheduler]afterDelay:2 schedule:^{
        
        LxPrintAnything(rac);
    }];
    
    //2. 每間隔多長時間做一件事
    [[RACSignal interval:1 onScheduler:[RACScheduler mainThreadScheduler]]subscribeNext:^(NSDate * date) {
        
        LxDBAnyVar(date);
    }];

}

#pragma mark 5.代理
- (void)delegateTest
{
    UIAlertView * alertView = [[UIAlertView alloc]initWithTitle:@"RAC" message:@"ReactiveCocoa" delegate:self cancelButtonTitle:@"Cancel" otherButtonTitles:@"Ensure", nil];
    
    [[self rac_signalForSelector:@selector(alertView:clickedButtonAtIndex:) fromProtocol:@protocol(UIAlertViewDelegate)] subscribeNext:^(RACTuple * tuple) {
        
        LxDBAnyVar(tuple);
        
        LxDBAnyVar(tuple.first);
        LxDBAnyVar(tuple.second);
        LxDBAnyVar(tuple.third);
    }];
    [alertView show];
    
    
    //  更簡單的方式:
    [[alertView rac_buttonClickedSignal]subscribeNext:^(id x) {
        
        LxDBAnyVar(x);
    }];

}

#pragma mark 6.KVO
/**    
 *  KVO
 *  RACObserveL:快速的監聽某個對象的某個屬性改變
 *  返回的是一個信號,對象的某個屬性改變的信號
    
- (void)test2 {
    [RACObserve(self.view, center) subscribeNext:^(id x) {
        NSLog(@"%@", x);
    }];
}
- (void)testAndtest2 // textField輸入的值賦值給label,監聽label文字改變,
    {
        
        RAC(self.label, text) = self.textField.rac_textSignal;
        [RACObserve(self.label, text) subscribeNext:^(id x) {
            NSLog(@"====label的文字變了");
        }];  
}      
     */
- (void)kvoTest
{
    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(CGSizeMake(CGRectGetWidth(self.view.frame), CGRectGetHeight(self.view.frame)));
    }];
    
    [RACObserve(scrollView, contentOffset) subscribeNext:^(id x) {
        
        LxDBAnyVar(x);
    }];
    
//  (好處:寫法簡單,keypath有代碼提示)
}
   
#pragma mark - concat 順序合并
// concat----- 使用需求:有兩部分數據:想讓上部分先執行,完了之后再讓下部分執行(都可獲取值)
- (void)concat {
    // 組合
    
    // 創建信號A
    RACSignal *signalA = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        // 發送請求
                NSLog(@"----發送上部分請求---afn");
        
        [subscriber sendNext:@"上部分數據"];
        [subscriber sendCompleted]; // 必須要調用sendCompleted方法!
        return nil;
    }];
    
    // 創建信號B,
    RACSignal *signalsB = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        // 發送請求
                NSLog(@"--發送下部分請求--afn");
        [subscriber sendNext:@"下部分數據"];
        return nil;
    }];
    
    
    // concat:按順序去鏈接
    //**-注意-**:concat,第一個信號必須要調用sendCompleted
    // 創建組合信號
    RACSignal *concatSignal = [signalA concat:signalsB];
    // 訂閱組合信號
    [concatSignal subscribeNext:^(id x) {
        NSLog(@"%@",x);
    }];
    
}
    
#pragma mark - zipWith 只有當兩個信號同時發出信號內容時
- (void)zipWith {
    //zipWith:把兩個信號壓縮成一個信號,只有當兩個信號同時發出信號內容時,并且把兩個信號的內容合并成一個元祖,才會觸發壓縮流的next事件。
    // 創建信號A
    RACSubject *signalA = [RACSubject subject];
    // 創建信號B
    RACSubject *signalB = [RACSubject subject];
    // 壓縮成一個信號
    // **-zipWith-**: 當一個界面多個請求的時候,要等所有請求完成才更新UI
    // 等所有信號都發送內容的時候才會調用
    RACSignal *zipSignal = [signalA zipWith:signalB];
    [zipSignal subscribeNext:^(id x) {
        NSLog(@"%@", x); //所有的值都被包裝成了元組
    }];
    
    // 發送信號 交互順序,元組內元素的順序不會變,跟發送的順序無關,而是跟壓縮的順序有關[signalA zipWith:signalB]---先是A后是B
    [signalA sendNext:@1];
    [signalB sendNext:@2];
    
}
    
#pragma mark - merge 多個信號合并成一個信號,任何一個信號有新值就會調用
// 任何一個信號請求完成都會被訂閱到
// merge:多個信號合并成一個信號,任何一個信號有新值就會調用
- (void)merge {
    // 創建信號A
    RACSubject *signalA = [RACSubject subject];
    // 創建信號B
    RACSubject *signalB = [RACSubject subject];
    //組合信號
    RACSignal *mergeSignal = [signalA merge:signalB];
    // 訂閱信號
    [mergeSignal subscribeNext:^(id x) {
        NSLog(@"%@", x);
    }];
    // 發送信號---交換位置則數據結果順序也會交換
    [signalB sendNext:@"下部分"];
    [signalA sendNext:@"上部分"];
}

#pragma mark - then
// then --- 使用需求:有兩部分數據:想讓上部分先進行網絡請求但是過濾掉數據,然后進行下部分的,拿到下部分數據
- (void)then {
    // 創建信號A
    RACSignal *signalA = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        // 發送請求
        NSLog(@"----發送上部分請求---afn");
        
        [subscriber sendNext:@"上部分數據"];
        [subscriber sendCompleted]; // 必須要調用sendCompleted方法!
        return nil;
    }];
    
    // 創建信號B,
    RACSignal *signalsB = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        // 發送請求
        NSLog(@"--發送下部分請求--afn");
        [subscriber sendNext:@"下部分數據"];
        [subscriber sendCompleted];
        return nil;
    }];
    // 創建組合信號
    // then;忽略掉第一個信號的所有值
    RACSignal *thenSignal = [signalA then:^RACSignal *{
        // 返回的信號就是要組合的信號
        return signalsB;
    }];
    
    // 訂閱信號
    [thenSignal subscribeNext:^(id x) {
        NSLog(@"%@", x);
    }];
    
}
    
#pragma mark stopwatch
- (void)stopwatch
{
    UILabel * label = ({
       
        UILabel * label = [[UILabel alloc]init];
        label.backgroundColor = [UIColor cyanColor];
        label;
    });
    [self.view addSubview:label];
    
    @weakify(self);
    
    [label mas_makeConstraints:^(MASConstraintMaker *make) {
        @strongify(self);
        
        make.size.mas_equalTo(CGSizeMake(240, 40));
        make.center.equalTo(self.view);
        
    }];
    
    RAC(label, text) = [[RACSignal interval:1 onScheduler:[RACScheduler mainThreadScheduler]] map:^NSString *(NSDate * date) {
        
        return date.description;
    }];
}
#pragma mark 8.RAC
- (void)RAC
{
    //button setBackgroundColor:forState:
    
    UIButton * button = [UIButton buttonWithType:UIButtonTypeCustom];
    [self.view addSubview:button];
    
    @weakify(self);
    
    [button mas_makeConstraints:^(MASConstraintMaker *make) {
        
        @strongify(self);
        make.size.mas_equalTo(CGSizeMake(180, 40));
        make.center.equalTo(self.view);
    }];
    
    /**
      RAC:把一個對象的某個屬性綁定一個信號,只要發出信號,就會把信號的內容給對象的屬性賦值
      給label的text屬性綁定了文本框改變的信號
        RAC(self.label, text) = self.textField.rac_textSignal;
         [self.textField.rac_textSignal subscribeNext:^(id x) {
             self.label.text = x;
         }];
     */
    RAC(button, backgroundColor) = [RACObserve(button, selected) map:^UIColor *(NSNumber * selected) {
        return [selected boolValue] ? [UIColor redColor] : [UIColor greenColor];
    }];
    
    //rac_valuesForKeyPath:
    [[button rac_signalForControlEvents:UIControlEventTouchUpInside]subscribeNext:^(UIButton * btn) {  
        btn.selected = !btn.selected;
    }];

    //tableView Cell
    // [[[button rac_signalForControlEvents:UIControlEventTouchUpInside] takeUntil:self.rac_prepareForReuseSignal] subscribeNext:^(UIButton *x) {
    // do other things
   //  }];

}

#pragma mark - 9.RACSignal   
/**
 RACSignal使用步驟:
 1.創建信號 + (RACSignal *)createSignal:(RACDisposable * (^)(id<RACSubscriber> subscriber))didSubscribe
 2.訂閱信號,才會激活信號. - (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock
 3.發送信號 - (void)sendNext:(id)value
 
 RACSignal底層實現:
 1.創建信號,首先把didSubscribe保存到信號中,還不會觸發。
 2.當信號被訂閱,也就是調用signal的subscribeNext:nextBlock
 2.2 subscribeNext內部會創建訂閱者subscriber,并且把nextBlock保存到subscriber中。
 2.1 subscribeNext內部會調用siganl的didSubscribe
 3.siganl的didSubscribe中調用[subscriber sendNext:@1];
 3.1 sendNext底層其實就是執行subscriber的nextBlock
 
 *  總結:
 我們完全可以用RACSubject代替代理/通知,確實方便許多
 這里我們點擊TwoViewController的pop的時候將字符串"ws"傳給了ViewController的button的title。
 步驟:
 // 1.創建信號
 RACSubject *subject = [RACSubject subject];
 
 // 2.訂閱信號
 [subject subscribeNext:^(id x) {
 // block:當有數據發出的時候就會調用
 // block:處理數據
 NSLog(@"%@",x);
 }];
 
 // 3.發送信號
 [subject sendNext:value];
 **注意:~~**
 RACSubject和RACReplaySubject的區別
 RACSubject必須要先訂閱信號之后才能發送信號,而RACReplaySubject可以先發送信號后訂閱.
 RACSubject 代碼中體現為:先走TwoViewController的sendNext,后走ViewController的subscribeNext訂閱
 RACReplaySubject 代碼中體現為:先走ViewController的subscribeNext訂閱,后走TwoViewController的sendNext
 可按實際情況各取所需。

 */

#pragma  mark repeat
- (void)repeatTest
{
    //repeat:
    [[[[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        
        [subscriber sendNext:@"rac"];
        [subscriber sendCompleted];
        
        return nil;
    }]delay:1]repeat]take:3] subscribeNext:^(id x) {
           
        LxDBAnyVar(x);
    } completed:^{
        
        LxPrintAnything(completed);
    }];

}
#pragma mark  throttle
- (void)throttle
{
    UITextField * textField = [[UITextField alloc]init];
    textField.backgroundColor = [UIColor cyanColor];
    [self.view addSubview:textField];
    
    @weakify(self);
    
    [textField mas_makeConstraints:^(MASConstraintMaker *make) {
        
        @strongify(self);
        make.size.mas_equalTo(CGSizeMake(180, 40));
        make.center.equalTo(self.view);
    }];
    //throttle 后面是個時間 表示rac_textSignal發送消息,0.3秒內沒有再次發送就會相應,若是0.3內又發送消息了,便會在新的信息處重新計時
    //distinctUntilChanged 表示兩個消息相同的時候,只會發送一個請求
    //ignore 表示如果消息和ignore后面的消息相同,則會忽略掉這條消息,不讓其發送
    
    [[[[[[textField.rac_textSignal throttle:0.3] distinctUntilChanged] ignore:@""] map:^id(id value) {
        
        return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
            
            //  network request
            [subscriber sendNext:value];
            [subscriber sendCompleted];
            
            return [RACDisposable disposableWithBlock:^{
                
                //  cancel request
            }];
        }];
    }]switchToLatest] subscribeNext:^(id x) {
        
        LxDBAnyVar(x);
    }];

    
}
#pragma mark takeOrSkip
- (void)takeOrSkip
{
    RACSignal * signal = [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        [subscriber sendNext:@"rac1"];
        [subscriber sendNext:@"rac2"];
        [subscriber sendNext:@"rac3"];
        [subscriber sendNext:@"rac4"];
        [subscriber sendCompleted];
        return nil;
    }]take:2];//Skip takeLast  takeUntil   takeWhileBlock:   skipWhileBlock:  skipUntilBlock:


    [signal subscribeNext:^(id x) {
        LxDBAnyVar(x);
    }];
}
#pragma mark timeOut
- (void)timeOut
{
    [[[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        
        [[RACScheduler mainThreadScheduler]afterDelay:3 schedule:^{
            
            [subscriber sendNext:@"rac"];
            [subscriber sendCompleted];
        }];
        
        return nil;
    }] timeout:2 onScheduler:[RACScheduler mainThreadScheduler]]
     subscribeNext:^(id x) {
         
         LxDBAnyVar(x);
     } error:^(NSError *error) {
         
         LxDBAnyVar(error);
     } completed:^{
         
         LxPrintAnything(completed);
     }];

}
#pragma mark startWith
- (void)startWith
{
    RACSignal * signal = [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        
        [subscriber sendNext:@"123"];//startWith:@"123"等同于這句話 也就是第一個發送,主要是位置
        [subscriber sendNext:@"rac"];
        [subscriber sendCompleted];
        return nil;
    }]startWith:@"123"];
    LxPrintAnything(start);
    //創建訂閱者
    [signal subscribeNext:^(id x) {
        LxDBAnyVar(x);
    }];

}
#pragma mark delay
- (void)delay
{
    //創建信號
    RACSignal * signal = [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        [subscriber sendNext:@"rac"];
        [subscriber sendCompleted];
        return nil;
    }]delay:2];
    LxPrintAnything(start);
    //創建訂閱者
    [signal subscribeNext:^(id x) {
        LxDBAnyVar(x);
    }];
    
}
#pragma mark map (映射)和filter
- (void)mapAndFilter
{
    UITextField * textField = ({
        UITextField * textField = [[UITextField alloc]init];
        textField.backgroundColor = [UIColor cyanColor];
        
        textField;
    });
    [self.view addSubview:textField];
    
    @weakify(self); //  __weak __typeof__(self) self_weak_ = self;
    
    [textField mas_makeConstraints:^(MASConstraintMaker *make) {
        
        @strongify(self);    // __strong __typeof__(self) self = self_weak_;
        make.size.mas_equalTo(CGSizeMake(180, 40));
        make.center.equalTo(self.view);
    }];

    [[[textField.rac_textSignal map:^id(NSString *text) {
        
       LxDBAnyVar(text);
        
        return @(text.length);
        
    }]filter:^BOOL(NSNumber *value) {
        
        return value.integerValue > 3;
        
    }] subscribeNext:^(id x) {
         LxDBAnyVar(x);
    }];

}
#pragma mark Signal
- (RACSignal *)createSignal
{
    return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
        
        RACDisposable * schedulerDisposable = [[RACScheduler mainThreadScheduler]afterDelay:2 schedule:^{
            
            if (arc4random()%10 > 1) {
                
                [subscriber sendNext:@"Login response"];
                [subscriber sendCompleted];
            }
            else {
                
                [subscriber sendError:[NSError errorWithDomain:@"LOGIN_ERROR_DOMAIN" code:444 userInfo:@{}]];
            }
        }];
        
        return [RACDisposable disposableWithBlock:^{
            
            [schedulerDisposable dispose];
        }];
    }];
}

#pragma mark 縮鍵盤
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    [self.view endEditing:YES];
}
@end

常見報錯

1.升級XCode7.3 后RAC報錯“Cannot create __weak reference in file using manual reference counting”解決辦法(http://blog.csdn.net/ouq68/article/details/51003876)

Please set ‘Weak References in Manual Retain Release:YES’.

2.[!] The Paopao [Debug] target overrides the PODS_ROOT build setting defined in `....

我想要使用 CocoaPods 中的設置,分別在我的項目中定義PODS_ROOT
和 Other Linker Flags 的地方,把他們的值用$(inherited) 替換掉,進入終端,執行

Demo 地址:RACDemo-master

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容