zoukankan      html  css  js  c++  java
  • ReactiveCocoa2实战

    转自无网不剩的博客

     
    之前已经写过两篇关于ReactiveCocoa(以下简称RAC)的文章了,但主要也是在阐述基本的概念和使用,这篇文章将会从实战的角度来看看RAC到底解决了哪些问题,带来了哪些方便,以及遇到的一些坑。
     
    概述
     
    为什么要使用RAC?
     
    一个怪怪的东西,从Demo看也没有让代码变得更好、更短,相反还造成理解上的困难,真的有必要去学它么?相信这是大多数人在接触RAC时的想法。RAC不是单一功能的模块,它是一个Framework,提供了一整套解决方案。其核心思想是「响应数据的变化」,在这个基础上有了Signal的概念,进而可以帮助减少状态变量(可以参考jspahrsummers的PPT),使用MVVM架构,统一的异步编程模型等等。
     
    为什么RAC更加适合编写Cocoa App?说这个之前,我们先来看下Web前端编程,因为有些相似之处。目前很火的AngularJS有一个很重要的特性:数据与视图绑定。就是当数据变化时,视图不需要额外的处理,便可正确地呈现最新的数据。而这也是RAC的亮点之一。RAC与Cocoa的编程模式,有点像AngularJS和jQuery。所以要了解RAC,需要先在观念上做调整。
     
    以下面这个Cell为例
    正常的写法可能是这样,很直观。
    1. - (void)configureWithItem:(HBItem *)item 
    2.     self.username.text = item.text; 
    3.     [self.avatarImageView setImageWithURL: item.avatarURL]; 
    4.     // 其他的一些设置 
    但如果用RAC,可能就是这样
    1. - (id)init 
    2.     if (self = [super init]) { 
    3.         @weakify(self); 
    4.         [RACObserve(self, viewModel) subscribeNext:^(HBItemViewModel *viewModel) { 
    5.             @strongify(self); 
    6.             self.username.text = viewModel.item.text; 
    7.             [self.avatarImageView setImageWithURL: viewModel.item.avatarURL]; 
    8.             // 其他的一些设置 
    9.         }]; 
    10.     } 
    也就是先把数据绑定,接下来只要数据有变化,就会自动响应变化。在这里,每次viewModel改变时,内容就会自动变成该viewModel的内容。
     
    Signal
     
    Signal是RAC的核心,为了帮助理解,画了这张简化图
     
    这里的数据源和sendXXX,可以理解为函数的参数和返回值。当Signal处理完数据后,可以向下一个Signal或Subscriber传送数据。可以看到上半部分的两个Signal是冷的(cold),相当于实现了某个函数,但该函数没有被调用。同时也说明了Signal可以被组合使用,比如RACSignal *signalB = [signalA map:^id(id x){return x}],或RACSignal *signalB = [signalA take:1]等等。
     
    当signal被subscribe时,就会处于热(hot)的状态,也就是该函数会被执行。比如上面的第二张图,首先signalA可能发了一个网络请求,拿到结果后,把数据通过sendNext方法传递到下一个signal,signalB可以根据需要做进一步处理,比如转换成相应的Model,转换完后再sendNext到subscriber,subscriber拿到数据后,再改变ViewModel,同时因为View已经绑定了ViewModel,所以拿到的数据会自动在View里呈现。
     
    还有,一个signal可以被多个subscriber订阅,这里怕显得太乱就没有画出来,但每次被新的subscriber订阅时,都会导致数据源的处理逻辑被触发一次,这很有可能导致意想不到的结果,需要注意一下。
     
    当数据从signal传送到subscriber时,还可以通过doXXX来做点事情,比如打印数据。
     
    通过这张图可以看到,这非常像中学时学的函数,比如 f(x) = y,某一个函数的输出又可以作为另一个函数的输入,比如 f(f(x)) = z,这也正是「函数响应式编程」(FRP)的核心。
     
    有些地方需要注意下,比如把signal作为local变量时,如果没有被subscribe,那么方法执行完后,该变量会被dealloc。但如果signal有被subscribe,那么subscriber会持有该signal,直到signal sendCompleted或sendError时,才会解除持有关系,signal才会被dealloc。
     
    RACCommand
     
    RACCommand是RAC很重要的组成部分,可以节省很多时间并且让你的App变得更Robust,这篇文章可以帮助你更深入的理解,这里简单做一下介绍。
     
    RACCommand 通常用来表示某个Action的执行,比如点击Button。它有几个比较重要的属性:executionSignals / errors / executing。
     
    1、executionSignals是signal of signals,如果直接subscribe的话会得到一个signal,而不是我们想要的value,所以一般会配合switchToLatest。
     
    2、errors。跟正常的signal不一样,RACCommand的错误不是通过sendError来实现的,而是通过errors属性传递出来的。
     
    3、executing表示该command当前是否正在执行。
     
    假设有这么个需求:当图片载入完后,分享按钮才可用。那么可以这样:
    1. RACSignal *imageAvailableSignal = [RACObserve(self, imageView.image) map:id^(id x){return x ? @YES : @NO}]; 
    2. self.shareButton.rac_command = [[RACCommand alloc] initWithEnabled:imageAvailableSignal signalBlock:^RACSignal *(id input) { 
    3.     // do share logic 
    4. }]; 
    除了与UIControl绑定之外,也可以手动执行某个command,比如双击图片点赞,就可以这么实现。
    1. // ViewModel.m 
    2. - (instancetype)init 
    3.     self = [super init]; 
    4.     if (self) { 
    5.         void (^updatePinLikeStatus)() = ^{ 
    6.             self.pin.likedCount = self.pin.hasLiked ? self.pin.likedCount - 1 : self.pin.likedCount + 1; 
    7.             self.pin.hasLiked = !self.pin.hasLiked; 
    8.         }; 
    9.          
    10.         _likeCommand = [[RACCommand alloc] initWithSignalBlock:^RACSignal *(id input) { 
    11.             // 先展示效果,再发送请求 
    12.             updatePinLikeStatus(); 
    13.             return [[HBAPIManager sharedManager] likePinWithPinID:self.pin.pinID]; 
    14.         }]; 
    15.          
    16.         [_likeCommand.errors subscribeNext:^(id x) { 
    17.             // 发生错误时,回滚 
    18.             updatePinLikeStatus(); 
    19.         }]; 
    20.     } 
    21.     return self; 
    22.  
    23. // ViewController.m 
    24. - (void)viewDidLoad 
    25.     [super viewDidLoad]; 
    26.     // ... 
    27.     @weakify(self); 
    28.     [RACObserve(self, viewModel.hasLiked) subscribeNex:^(id x){ 
    29.         @strongify(self); 
    30.         self.pinLikedCountLabel.text = self.viewModel.likedCount; 
    31.         self.likePinImageView.image = [UIImage imageNamed:self.viewModel.hasLiked ? @"pin_liked" : @"pin_like"]; 
    32.     }]; 
    33.      
    34.     UITapGestureRecognizer *tapGesture = [[UITapGestureRecognizer alloc] init]; 
    35.     tapGesture.numberOfTapsRequired = 2; 
    36.     [[tapGesture rac_gestureSignal] subscribeNext:^(id x) { 
    37.         [self.viewModel.likeCommand execute:nil]; 
    38.     }]; 
    再比如某个App要通过Twitter登录,同时允许取消登录,就可以这么做 (source)
    1. _twitterLoginCommand = [[RACCommand alloc] initWithSignalBlock:^(id _) { 
    2.       @strongify(self); 
    3.       return [[self  
    4.           twitterSignInSignal]  
    5.           takeUntil:self.cancelCommand.executionSignals]; 
    6.     }]; 
    7.  
    8. RAC(self.authenticatedUser) = [self.twitterLoginCommand.executionSignals switchToLatest]; 
     
    常用的模式
     
    map + switchToLatest
     
    switchToLatest: 的作用是自动切换signal of signals到最后一个,比如之前的command.executionSignals就可以使用switchToLatest:。
     
    map:的作用很简单,对sendNext的value做一下处理,返回一个新的值。
     
    如果把这两个结合起来就有意思了,想象这么个场景,当用户在搜索框输入文字时,需要通过网络请求返回相应的hints,每当文字有变动时,需要取消上一次的请求,就可以使用这个配搭。这里用另一个Demo,简单演示一下
    1. NSArray *pins = @[@172230988, @172230947, @172230899, @172230777, @172230707]; 
    2. __block NSInteger index = 0; 
    3.  
    4. RACSignal *signal = [[[[RACSignal interval:0.1 onScheduler:[RACScheduler scheduler]] 
    5.                         take:pins.count] 
    6.                         map:^id(id value) { 
    7.                             return [[[HBAPIManager sharedManager] fetchPinWithPinID:[pins[index++] intValue]] doNext:^(id x) { 
    8.                                 NSLog(@"这里只会执行一次"); 
    9.                             }]; 
    10.                         }] 
    11.                         switchToLatest]; 
    12.  
    13. [signal subscribeNext:^(HBPin *pin) { 
    14.     NSLog(@"pinID:%d", pin.pinID); 
    15. } completed:^{ 
    16.     NSLog(@"completed"); 
    17. }]; 
    18.  
    19. // output 
    20. // 2014-06-05 17:40:49.851 这里只会执行一次 
    21. // 2014-06-05 17:40:49.851 pinID:172230707 
    22. // 2014-06-05 17:40:49.851 completed 
     
    takeUntil
     
    takeUntil:someSignal 的作用是当someSignal sendNext时,当前的signal就sendCompleted,someSignal就像一个拳击裁判,哨声响起就意味着比赛终止。
     
    它的常用场景之一是处理cell的button的点击事件,比如点击Cell的详情按钮,需要push一个VC,就可以这样:
    1. [[[cell.detailButton 
    2.     rac_signalForControlEvents:UIControlEventTouchUpInside] 
    3.     takeUntil:cell.rac_prepareForReuseSignal] 
    4.     subscribeNext:^(id x) { 
    5.         // generate and push ViewController 
    6. }]; 
    如果不加takeUntil:cell.rac_prepareForReuseSignal,那么每次Cell被重用时,该button都会被addTarget:selector。
     
    替换Delegate
     
    出现这种需求,通常是因为需要对Delegate的多个方法做统一的处理,这时就可以造一个signal出来,每次该Delegate的某些方法被触发时,该signal就会sendNext。
    1. @implementation UISearchDisplayController (RAC) 
    2. - (RACSignal *)rac_isActiveSignal { 
    3.     self.delegate = self; 
    4.     RACSignal *signal = objc_getAssociatedObject(self, _cmd); 
    5.     if (signal != nil) return signal; 
    6.      
    7.     /* Create two signals and merge them */ 
    8.     RACSignal *didBeginEditing = [[self rac_signalForSelector:@selector(searchDisplayControllerDidBeginSearch:)  
    9.                                         fromProtocol:@protocol(UISearchDisplayDelegate)] mapReplace:@YES]; 
    10.     RACSignal *didEndEditing = [[self rac_signalForSelector:@selector(searchDisplayControllerDidEndSearch:)  
    11.                                       fromProtocol:@protocol(UISearchDisplayDelegate)] mapReplace:@NO]; 
    12.     signal = [RACSignal merge:@[didBeginEditing, didEndEditing]]; 
    13.      
    14.      
    15.     objc_setAssociatedObject(self, _cmd, signal, OBJC_ASSOCIATION_RETAIN_NONATOMIC); 
    16.     return signal; 
    17. @end 
    代码源于此文
     
    使用ReactiveViewModel的didBecomActiveSignal
     
    ReactiveViewModel是另一个project, 后面的MVVM中会讲到,通常的做法是在VC里设置VM的active属性(RVMViewModel自带该属性),然后在VM里subscribeNext didBecomActiveSignal,比如当Active时,获取TableView的最新数据。
     
    RACSubject的使用场景
     
    一般不推荐使用RACSubject,因为它过于灵活,滥用的话容易导致复杂度的增加。但有一些场景用一下还是比较方便的,比如ViewModel的errors。
     
    ViewModel一般会有多个RACCommand,那这些commands如果出现error了该如何处理呢?比较方便的方法如下:
    1. // HBCViewModel.h 
    2.  
    3. #import "RVMViewModel.h" 
    4.  
    5. @class RACSubject; 
    6.  
    7. @interface HBCViewModel : RVMViewModel 
    8. @property (nonatomic) RACSubject *errors; 
    9. @end 
    10.  
    11.  
    12.  
    13. // HBCViewModel.m 
    14.  
    15. #import "HBCViewModel.h" 
    16. #import <ReactiveCocoa.h> 
    17.  
    18. @implementation HBCViewModel 
    19.  
    20. - (instancetype)init 
    21.     self = [super init]; 
    22.     if (self) { 
    23.         _errors = [RACSubject subject]; 
    24.     } 
    25.     return self; 
    26.  
    27. - (void)dealloc 
    28.     [_errors sendCompleted]; 
    29. @end 
    30.  
    31. // Some Other ViewModel inherit HBCViewModel 
    32.  
    33. - (instancetype)init 
    34.     _fetchLatestCommand = [RACCommand alloc] initWithSignalBlock:^RACSignal *(id input){ 
    35.         // fetch latest data 
    36.     }]; 
    37.  
    38.     _fetchMoreCommand = [RACCommand alloc] initWithSignalBlock:^RACSignal *(id input){ 
    39.         // fetch more data 
    40.     }]; 
    41.  
    42.     [self.didBecomeActiveSignal subscribeNext:^(id x) { 
    43.         [_fetchLatestCommand execute:nil]; 
    44.     }]; 
    45.      
    46.     [[RACSignal 
    47.         merge:@[ 
    48.                 _fetchMoreCommand.errors, 
    49.                 _fetchLatestCommand.errors 
    50.                 ]] subscribe:self.errors]; 
    51.  
     
    rac_signalForSelector
     
    rac_signalForSelector: 这个方法会返回一个signal,当selector执行完时,会sendNext,也就是当某个方法调用完后再额外做一些事情。用在category会比较方便,因为Category重写父类的方法时,不能再通过[super XXX]来调用父类的方法,当然也可以手写Swizzle来实现,不过有了rac_signalForSelector:就方便多了。
     
    rac_signalForSelector: fromProtocol: 可以直接实现对protocol的某个方法的实现(听着有点别扭呢),比如,我们想实现UIScrollViewDelegate的某些方法,可以这么写
    1. [[self rac_signalForSelector:@selector(scrollViewDidEndDecelerating:) fromProtocol:@protocol(UIScrollViewDelegate)] subscribeNext:^(RACTuple *tuple) { 
    2.     // do something 
    3. }]; 
    4.  
    5. [[self rac_signalForSelector:@selector(scrollViewDidScroll:) fromProtocol:@protocol(UIScrollViewDelegate)] subscribeNext:^(RACTuple *tuple) { 
    6.     // do something 
    7. }]; 
    8.  
    9. self.scrollView.delegate = nil; 
    10. self.scrollView.delegate = self; 
    注意,这里的delegate需要先设置为nil,再设置为self,而不能直接设置为self,如果self已经是该scrollView的Delegate的话。
     
    有时,我们想对selector的返回值做一些处理,但很遗憾RAC不支持,如果真的有需要的话,可以使用Aspects
     
    MVVM
     
    这是一个大话题,如果有耐心,且英文还不错的话,可以看一下Cocoa Samurai的这两篇文章。PS: Facebook Paper就是基于MVVM构建的。
     
    MVVM是Model-View-ViewModel的简称,它们之间的关系如下
    可以看到View(其实是ViewController)持有ViewModel,这样做的好处是ViewModel更加独立且可测试,ViewModel里不应包含任何View相关的元素,哪怕换了一个View也能正常工作。而且这样也能让View/ViewController「瘦」下来。
     
    ViewModel主要做的事情是作为View的数据源,所以通常会包含网络请求。
     
    或许你会疑惑,ViewController哪去了?在MVVM的世界里,ViewController已经成为了View的一部分。它的主要职责是将VM与View绑定、响应VM数据的变化、调用VM的某个方法、与其他的VC打交道。
     
    而RAC为MVVM带来很大的便利,比如RACCommand, UIKit的RAC Extension等等。使用MVVM不一定能减少代码量,但能降低代码的复杂度。
     
    以下面这个需求为例,要求大图滑动结束时,底部的缩略图滚动到对应的位置,并高亮该缩略图;同时底部的缩略图被选中时,大图也要变成该缩略图的大图。
    我的思路是横向滚动的大图是一个collectionView,该collectionView是当前页面VC的一个property。底部可以滑动的缩略图是一个childVC的collectionView,这两个collectionView共用一套VM,并且各自RACObserve感兴趣的property。
     
    比如大图滑到下一页时,会改变VM的indexPath属性,而底部的collectionView所在的VC正好对该indexPath感兴趣,只要indexPath变化就滚动到相应的Item
    1. // childVC 
    2.  
    3. - (void)viewDidLoad 
    4.     [super viewDidLoad]; 
    5.  
    6.     @weakify(self); 
    7.     [RACObserve(self, viewModel.indexPath) subscribeNext:^(NSNumber *index) { 
    8.         @strongify(self); 
    9.         [self scrollToIndexPath]; 
    10.     }]; 
    11.  
    12. - (void)scrollToIndexPath 
    13.     if (self.collectionView.subviews.count) { 
    14.         NSIndexPath *indexPath = self.viewModel.indexPath; 
    15.         [self.collectionView scrollToItemAtIndexPath:indexPath atScrollPosition:UICollectionViewScrollPositionCenteredHorizontally animated:YES]; 
    16.         [self.collectionView.subviews enumerateObjectsUsingBlock:^(UIView *view, NSUInteger idx, BOOL *stop) { 
    17.             view.layer.borderWidth = 0; 
    18.         }]; 
    19.         UIView *view = [self.collectionView cellForItemAtIndexPath:indexPath]; 
    20.         view.layer.borderWidth = kHBPinsNaviThumbnailPadding; 
    21.         view.layer.borderColor = [UIColor whiteColor].CGColor; 
    22.     } 
    当点击底部的缩略图时,上面的大图也要做出变化,也同样可以通过RACObserve indexPath来实现
    1. // PinsViewController.m 
    2. - (void)viewDidLoad 
    3.     [super viewDidLoad]; 
    4.     @weakify(self); 
    5.     [[RACObserve(self, viewModel.indexPath) 
    6.         skip:1] 
    7.         subscribeNext:^(NSIndexPath *indexPath) { 
    8.             @strongify(self); 
    9.             [self.collectionView scrollToItemAtIndexPath:indexPath atScrollPosition:UICollectionViewScrollPositionCenteredHorizontally animated:YES]; 
    10.     }]; 
    这里有一个小技巧,当Cell里的元素比较复杂时,我们可以给Cell也准备一个ViewModel,这个CellViewModel可以由上一层的ViewModel提供,这样Cell如果需要相应的数据,直接跟CellViewModel要即可,CellViewModel也可以包含一些command,比如likeCommand。假如点击Cell时,要做一些处理,也很方便。
    1. // CellViewModel已经在ViewModel里准备好了 
    2. - (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath 
    3.     HBPinsCell *cell = [collectionView dequeueReusableCellWithReuseIdentifier:cellIdentifier forIndexPath:indexPath]; 
    4.     cell.viewModel = self.viewModel.cellViewModels[indexPath.row]; 
    5.     return cell; 
    6.  
    7. - (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath 
    8.     HBCellViewModel *cellViewModel = self.viewModel.cellViewModels[indexPath.row]; 
    9.     // 对cellViewModel执行某些操作,因为Cell已经与cellViewModel绑定,所以cellViewModel的改变也会反映到Cell上 
    10.     // 或拿到cellViewModel的数据来执行某些操作 
     
    ViewModel中signal, property, command的使用
     
    初次使用RAC+MVVM时,往往会疑惑,什么时候用signal,什么时候用property,什么时候用command?
     
    一般来说可以使用property的就直接使用,没必要再转换成signal,外部RACObserve即可。使用signal的场景一般是涉及到多个property或多个signal合并为一个signal。command往往与UIControl/网络请求挂钩。
     
    常见场景的处理
     
    检查本地缓存,如果失效则去请求网络数据并缓存到本地
     
    1. - (RACSignal *)loadData { 
    2.     return [[RACSignal  
    3.         createSignal:^(id<RACSubscriber> subscriber) { 
    4.             // If the cache is valid then we can just immediately send the  
    5.             // cached data and be done. 
    6.             if (self.cacheValid) { 
    7.                 [subscriber sendNext:self.cachedData]; 
    8.                 [subscriber sendCompleted]; 
    9.             } else { 
    10.                 [subscriber sendError:self.staleCacheError]; 
    11.             } 
    12.         }]  
    13.         // Do the subscription work on some random scheduler, off the main  
    14.         // thread. 
    15.         subscribeOn:[RACScheduler scheduler]]; 
    16.  
    17. - (void)update { 
    18.     [[[[self  
    19.         loadData] 
    20.         // Catch the error from -loadData. It means our cache is stale. Update 
    21.         // our cache and save it. 
    22.         catch:^(NSError *error) { 
    23.             return [[self updateCachedData] doNext:^(id data) { 
    24.                 [self cacheData:data]; 
    25.             }]; 
    26.         }]  
    27.         // Our work up until now has been on a background scheduler. Get our  
    28.         // results delivered on the main thread so we can do UI work. 
    29.         deliverOn:RACScheduler.mainThreadScheduler] 
    30.         subscribeNext:^(id data) { 
    31.             // Update your UI based on `data`. 
    32.  
    33.             // Update again after `updateInterval` seconds have passed. 
    34.             [[RACSignal interval:updateInterval] take:1] subscribeNext:^(id _) { 
    35.                 [self update]; 
    36.             }]; 
    37.         }];  
     
    检测用户名是否可用
     
    1. - (void)setupUsernameAvailabilityChecking { 
    2.     RAC(self, availabilityStatus) = [[[RACObserve(self.userTemplate, username) 
    3.                                       throttle:kUsernameCheckThrottleInterval] //throttle表示interval时间内如果有sendNext,则放弃该nextValue 
    4.                                       map:^(NSString *username) { 
    5.                                           if (username.length == 0) return [RACSignal return:@(UsernameAvailabilityCheckStatusEmpty)]; 
    6.                                           return [[[[[FIBAPIClient sharedInstance] 
    7.                                                 getUsernameAvailabilityFor:username ignoreCache:NO] 
    8.                                               map:^(NSDictionary *result) { 
    9.                                                   NSNumber *existsNumber = result[@"exists"]; 
    10.                                                   if (!existsNumber) return @(UsernameAvailabilityCheckStatusFailed); 
    11.                                                   UsernameAvailabilityCheckStatus status = [existsNumber boolValue] ? UsernameAvailabilityCheckStatusUnavailable : UsernameAvailabilityCheckStatusAvailable; 
    12.                                                   return @(status); 
    13.                                               }] 
    14.                                              catch:^(NSError *error) { 
    15.                                                   return [RACSignal return:@(UsernameAvailabilityCheckStatusFailed)]; 
    16.                                               }] startWith:@(UsernameAvailabilityCheckStatusChecking)]; 
    17.                                       }] 
    18.                                       switchToLatest]; 
    可以看到这里也使用了map + switchToLatest模式,这样就可以自动取消上一次的网络请求。
     
    startWith的内部实现是concat,这里表示先将状态置为checking,然后再根据网络请求的结果设置状态。
     
    使用takeUntil:来处理Cell的button点击
     
    这个上面已经提到过了。
     
    token过期后自动获取新的
     
    开发APIClient时,会用到AccessToken,这个Token过一段时间会过期,需要去请求新的Token。比较好的用户体验是当token过期后,自动去获取新的Token,拿到后继续上一次的请求,这样对用户是透明的。
    1. RACSignal *requestSignal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { 
    2.         // suppose first time send request, access token is expired or invalid 
    3.         // and next time it is correct. 
    4.         // the block will be triggered twice. 
    5.         static BOOL isFirstTime = 0; 
    6.         NSString *url = @"http://httpbin.org/ip"; 
    7.         if (!isFirstTime) { 
    8.             url = @"http://nonexists.com/error"; 
    9.             isFirstTime = 1; 
    10.         } 
    11.         NSLog(@"url:%@", url); 
    12.         [[AFHTTPRequestOperationManager manager] GET:url parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) { 
    13.             [subscriber sendNext:responseObject]; 
    14.             [subscriber sendCompleted]; 
    15.         } failure:^(AFHTTPRequestOperation *operation, NSError *error) { 
    16.             [subscriber sendError:error]; 
    17.         }]; 
    18.         return nil; 
    19.     }]; 
    20.      
    21.     self.statusLabel.text = @"sending request..."; 
    22.     [[requestSignal catch:^RACSignal *(NSError *error) { 
    23.         self.statusLabel.text = @"oops, invalid access token"; 
    24.          
    25.         // simulate network request, and we fetch the right access token 
    26.         return [[RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) { 
    27.             double delayInSeconds = 1.0; 
    28.             dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC)); 
    29.             dispatch_after(popTime, dispatch_get_main_queue(), ^(void){ 
    30.                 [subscriber sendNext:@YES]; 
    31.                 [subscriber sendCompleted]; 
    32.             }); 
    33.             return nil; 
    34.         }] concat:requestSignal]; 
    35.     }] subscribeNext:^(id x) { 
    36.         if ([x isKindOfClass:[NSDictionary class]]) { 
    37.             self.statusLabel.text = [NSString stringWithFormat:@"result:%@", x[@"origin"]]; 
    38.         } 
    39.     } completed:^{ 
    40.         NSLog(@"completed"); 
    41.     }]; 
     
    注意事项
     
    RAC我自己感觉遇到的几个难点是: 1) 理解RAC的理念。 2) 熟悉常用的API。3) 针对某些特定的场景,想出比较合理的RAC处理方式。不过看多了,写多了,想多了就会慢慢适应。下面是我在实践过程中遇到的一些小坑。
     
    ReactiveCocoaLayout
     
    有时Cell的内容涉及到动态的高度,就会想到用Autolayout来布局,但RAC已经为我们准备好了ReactiveCocoaLayout,所以我想不妨就拿来用一下。
     
    ReactiveCocoaLayout的使用好比「批地」和「盖房」,先通过insetWidth:height:nullRect从某个View中划出一小块,拿到之后还可以通过divideWithAmount:padding:fromEdge 再分成两块,或sliceWithAmount:fromEdge再分出一块。这些方法返回的都是signal,所以可以通过RAC(self.view, frame) = someRectSignal 这样来实现绑定。但在实践中发现性能不是很好,多批了几块地就容易造成主线程卡顿。
     
    所以ReactiveCocoaLayout最好不用或少用。
     
    调试
    刚开始写RAC时,往往会遇到这种情况,满屏的调用栈信息都是RAC的,要找出真正出现问题的地方不容易。曾经有一次在使用[RACSignal combineLatest: reduce:^id{}]时,忘了在Block里返回value,而Xcode也没有提示warning,然后就是莫名其妙地挂起了,跳到了汇编上,也没有调用栈信息,这时就只能通过最古老的注释代码的方式来找到问题的根源。
     
    不过写多了之后,一般不太会犯这种低级错误。
     
    strongify / weakify dance
     
    因为RAC很多操作都是在Block中完成的,这块最常见的问题就是在block直接把self拿来用,造成block和self的retain cycle。所以需要通过@strongify和@weakify来消除循环引用。
     
    有些地方很容易被忽略,比如RACObserve(thing, keypath),看上去并没有引用self,所以在subscribeNext时就忘记了weakify/strongify。但事实上RACObserve总是会引用self,即使target不是self,所以只要有RACObserve的地方都要使用weakify/strongify。
     
    小结
     
    以上是我在做花瓣客户端和side project时总结的一些经验,但愿能带来一些帮助,有误的地方也欢迎指正和探讨。
     
    推荐一下jspahrsummers的这个project,虽然是用RAC3.0写的,但很多理念也可以用到RAC2上面。
     
    最后感谢Github的iOS工程师们,感谢你们带来了RAC,以及在Issues里的耐心解答。
  • 相关阅读:
    最接近的三数之和(给定一个包括 n 个整数的数组 nums 和 一个目标值 target。找出 nums 中的三个整数, 使得它们的和与 target 最接近。返回这三个数的和)
    在排序数组中查找元素的第一个和最后一个位置(给定一个按照升序排列的整数数组 nums,和一个目标值 target。找出给定目标值在数组中的开始位置和结束位置。)
    查找常用字符(给定仅有小写字母组成的字符串数组 A,返回列表中的每个字符串中都显示的全部字符(包括重复字符)组成的列表。例如,如果一个字符在每个字符串中出现 3 次,但不是 4 次,则需要在最终答案中包含该字符 3 次。)
    三数之和等于0问题优化
    线性表之数组实现
    LeetCode竞赛题:笨阶乘(我们设计了一个笨阶乘 clumsy:在整数的递减序列中,我们以一个固定顺序的操作符序列来依次替换原有的乘法操作符:乘法(*),除法(/),加法(+)和减法(-)。)
    LeetCode竞赛题:K 次取反后最大化的数组和(给定一个整数数组 A,我们只能用以下方法修改该数组:我们选择某个个索引 i 并将 A[i] 替换为 -A[i],然后总共重复这个过程 K 次。)
    最短路径(给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。 说明:每次只能向下或者向右移动一步。)
    不同路径II(一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为“Start” )。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为“Finish”)。 现在考虑网格中有障碍物。那么从左上角到右下角将会有多少条不同的路径?网格中的障碍物和空位置分别用 1 和 0 来表示。)
    不同路径(一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为“Start” )。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为“Finish”)。 问总共有多少条不同的路径?)
  • 原文地址:https://www.cnblogs.com/PengFei-N/p/4747967.html
Copyright © 2011-2022 走看看