使用 ReactiveCocoa (下文將用 RAC 簡稱)有一段時間了,寫點東西記錄一下心得,權(quán)當分享。我的知識大多都是從開源社區(qū),或者其他人的分享處得來,知識如果不分享,那將沒有意義。所以我很希望自己的經(jīng)驗也能夠幫助一些新人,盡早走出新手村。
為什么要用 RAC ?
優(yōu)點:對個人而言,RAC 能節(jié)省代碼量,項目的結(jié)構(gòu)更漂亮,邏輯更清晰。 節(jié)省代碼量 ,就能更專注于其他工作,而不需要書寫一些繁雜的與業(yè)務(wù)邏輯無關(guān)的代碼實現(xiàn)。節(jié)省代碼量另一種意義上來說就是等于 珍惜生命 。隨著代碼量的減少,代碼結(jié)構(gòu)也會變得更清晰。
看起來很棒,這東西沒有缺點?當然有,如果說人們對某樣?xùn)|西只談優(yōu)點不談缺點,那一定是在耍流氓……
缺點:函數(shù)式編程的思想對新手不友好,可讀性對新手不友好,有一定的學(xué)習(xí)時間成本。
總的來說,好處還是占很大比例,我不打算講解底層概念,寫得很清楚的大有人在。我只想通過一些簡單的盡可能精簡的小例子拋磚引玉,讓新手能快速上手,直接能看到使用RAC之后的好處,來決定是否使用RAC來改善工作流程。
安裝RAC
簡述一下安裝步驟
首先利用CocoaPods安裝RAC。
2.在podfile文件中加入 pod 'ReactiveCocoa', '~> 2.5'(Objective-C版本,指定2.5版本)。
3.終端執(zhí)行pod update執(zhí)行安裝。
Demo,項目中的常用方法:
UIButton 事件的點擊
// 生成按鈕
- (void)setupButton{
UIButton *button = [[UIButton alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
[button setTitle:@"按鈕" forState:UIControlStateNormal];
[self.view addSubview:button];
[button addTarget:self action:@selector(click) forControlEvents:UIControlEventTouchUpInside];
}
// 點擊事件
- (void)click{
NSLog(@"按鈕點擊了");
}
可以看到,這是生成一個按鈕需要的代碼,如果某個按鈕要遷移到另一個頁面,或者現(xiàn)在需要刪除某個按鈕,就要找到兩個地方才能刪除干凈,我個人覺得,它很分散,UI和事件分開,界面中只有一兩個按鈕那還算好找,實際項目中的代碼量會比較多,找起來非常非常會心累。用RAC怎么實現(xiàn)呢?嗯,這樣:
- (void)setupButton{
// UI
UIButton *button = [[UIButton alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
[button setTitle:@"按鈕" forState:UIControlStateNormal];
[self.view addSubview:button];
// 事件
[[button rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(id x) {
NSLog(@"按鈕點擊了");
}];
}
非???,UI和事件終于聚集在一起了,復(fù)制或者刪除將會很方便,看到一個按鈕就能緊跟著看到它的實現(xiàn),減少了心智負擔(dān)。
UITextField 獲取每一次輸入的數(shù)據(jù)
- (void)setupUITextField{
// UI
UITextField *TextField = [[UITextField alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
[self.view addSubview:TextField];
// 事件
[TextField.rac_textSignal subscribeNext:^(NSString *x) {
NSLog(@"當前輸入框輸入字符為:%@", x);
}];
}
每次有字符輸入的時候,都會觸發(fā)Block中的內(nèi)容,非常方便就能獲取到數(shù)據(jù)。
UITableView 監(jiān)聽滾動偏移值
- (void)setupUITextField{
// UI
UITableView *tableView = [[UITableView alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
[self.view addSubview:tableView];
// 事件
[RACObserve(tableView, contentOffset) subscribeNext:^(id x) {
CGPoint offset = [x CGPointValue];
NSLog(@"tableView的偏移值發(fā)生改變,偏移值為:x:%f,y:%f", offset.x, offset.y);
}];
}
免去了寫KVO的繁雜代碼,用RAC來監(jiān)聽TableView的偏移值,訂閱到的數(shù)據(jù)是對象類型,要轉(zhuǎn)換為CGPoint結(jié)構(gòu)體才能使用。
利用RAC避免回調(diào)地獄
回調(diào)地獄(多層回調(diào)嵌套)是很可怕的事情,假設(shè)我們有一個頁面,需要依賴三個數(shù)據(jù),三個數(shù)據(jù)同時下載完畢后再刷新頁面。不太好的寫法如下:
- (void)downData{
__block id a, b ,c;
[self download:^(NSData *data) {
NSLog(@"數(shù)據(jù)a下載完成");
a = data;
[self download:^(NSData *data) {
NSLog(@"數(shù)據(jù)b下載完成");
b = data;
[self download:^(NSData *data) {
NSLog(@"數(shù)據(jù)c下載完成");
c = data;
[self setupView:@[a, b, c]];
}];
}];
}];
}
- (void)setupView:(NSArray *)datas{
NSLog(@"利用abc數(shù)據(jù)刷新頁面");
}
看起來非常多層級,比較混亂,接下來要使用RAC來優(yōu)化代碼,讓層級清晰合理一些。
- (void)downData {
__block id a, b, c;
// 生成三個信號
RACSubject *signal1 = [RACSubject subject];
RACSubject *signal2 = [RACSubject subject];
RACSubject *signal3 = [RACSubject subject];
// 合并信號,三個信號都觸發(fā)后,才激活 subscribeNext Block
[[RACSignal combineLatest:@[ signal1, signal2, signal3 ]
reduce:^id {
return @"";
}] subscribeNext:^(id x) {
NSLog(@"全部數(shù)據(jù)下載完畢,刷新UI");
[self setupView:@[ a, b, c ]];
}];
[self download:^(NSData *data) {
NSLog(@"數(shù)據(jù)a下載完成");
a = data;
[signal1 sendNext:nil];
}];
[self download:^(NSData *data) {
NSLog(@"數(shù)據(jù)b下載完成");
b = data;
[signal2 sendNext:nil];
}];
[self download:^(NSData *data) {
NSLog(@"數(shù)據(jù)c下載完成");
c = data;
[signal3 sendNext:nil];
}];
}
- (void)setupView:(NSArray *)datas {
NSLog(@"利用abc數(shù)據(jù)刷新頁面");
}
這樣代碼就清晰多了,以上一些方法都是常用的部分方法,簡單列出自己的使用經(jīng)驗,給新人們做個參考。時間有限,寫得有點匆忙,有什么問題,請給我評論,或者留言。我會盡心盡力幫助每一個新人。好了,下次再見~
拖延癥一犯,不知何年何月……