block 和 ReactiveCocoa的理解(一)

1. block的最大用处:回调(callback)

虽然接触iOS已经8个月了,block 作为Objective C中对于回调(callback)的实现,理解起来还是有点模棱两可。在《Pro Multithreading and Memory Management for iOS and OS X》书中,Kazuki Sakamoto 对block的定义是:

拥有自动变量(可以在block声明的语义环境里捕捉变量的状态)的匿名(使函数体(code)成为和数据(data)一样的一等公民,作为函数调用时输入的实参(argument))函数。

我发现,在大部分应用block的场景,这样的定义不够sharp,使人模棱两可。我用我自己的理解重定义block:

拥有自动变量(可以在block声明的语义环境里捕捉变量的状态)的匿名(使函数体(code)成为和数据(data)一样的一等公民,作为函数调用时输入的实参(argument))回调(callback,等待被调用)函数。

回调(callback)两个字把block在大部分场景应用时的扮演的角色给准确描述。下面我举一个简单的例子来解释回调。NSArray的实例方法 *- enumerateObjectsUsingBlock:(void (^ )(id obj, NSInteger idx, BOOL *stop))block 是一个用block实现数组枚举的方法。下面是该方法的简单应用:

NSArray *cities = @[@"Beijing", @"Shanghai", @"Guangzhou", @"Shenzhen",@"Hong Kong"];
[cities enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL * stop) {
    NSLog(@"index: %lu, obj: %@, stop: %d",(unsigned long)idx,obj,*stop);
    if(idx == 3){
        *stop = YES;
    }
}];

//这是一个再简单不过的数组枚举。输出为:

2016-01-06 22:52:18.143 Chapter00_Block implmenetaion[18596:780185] index: 0, obj: Beijing, stop: 0 
2016-01-06 22:52:18.144 Chapter00_Block implmenetaion[18596:780185] index: 1, obj: Shanghai, stop: 0 
2016-01-06 22:52:18.144 Chapter00_Block implmenetaion[18596:780185] index: 2, obj: Guangzhou, stop: 0 
2016-01-06 22:52:18.144 Chapter00_Block implmenetaion[18596:780185] index: 3, obj: Shenzhen, stop: 0

那么,如何理解上面block的新定义里的回调呢。由于apple 自己里大部分类的实现是保密的,我们来自己定义一个方法来简单实现下这个- enumerateObjectsUsingBlock:方法的功能。首先来给NSArray添加一个名为Enumeration的category,加入以下代码。

NSArray+Enumeration.h

#import <Foundation/Foundation.h> 
@interface NSArray (Enumeration) 
- (void)customizedEnumerateWithBlock:(void(^)(id obj, NSInteger index, BOOL *stop))block; 
@end

NSArray+Enumeration.m

#import "NSArray+Enumeration.h" 
@implementation NSArray (Enumeration) 
-(void)customizedEnumerateWithBlock:(void (^)(id, NSInteger, BOOL *))block{ 
    BOOL stop = false; for(int i = 0; i < self.count; ++i){ 
        block(i,[self objectAtIndex:i],&stop); 
        if(stop) break;
    }
 }  
@end

如果你把cities的数组枚举方法换成这个,那么输出是一样的。

NSArray *cities = @[@"Beijing", @"Shanghai", @"Guangzhou", @"Shenzhen", @"Hong Kong"];
[cities customizedEnumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL * stop) {
    NSLog(@"index: %lu, obj: %@, stop: %d",(unsigned long)idx,obj,*stop);
    if(idx == 3){
        *stop = YES;
    }
}];

//输出为:

2016-01-06 22:52:18.143 Chapter00_Block implmenetaion[18596:780185] index: 0, obj: Beijing, stop: 0 
2016-01-06 22:52:18.144 Chapter00_Block implmenetaion[18596:780185] index: 1, obj: Shanghai, stop: 0 
2016-01-06 22:52:18.144 Chapter00_Block implmenetaion[18596:780185] index: 2, obj: Guangzhou, stop: 0 
2016-01-06 22:52:18.144 Chapter00_Block implmenetaion[18596:780185] index: 3, obj: Shenzhen, stop: 0

由此可见,我们自己实现的数组枚举能实现原本的数组枚举方法。在- customizedEnumerateWithBlock: 的实现里,回调了block这个函数。因此若有方法里有block体作为参数,则意味著这个方法在调用其他函数的同时,这个block在等待将来被调用。那么该block什么时候被调用呢(这里是for 循环语句里),在实际应用block场景里,block的调用更多地是简化网络请求,delegate等。

因此“回调”是block我个人认为最大的用处。

2. ReactiveCocoa的subscription原理

作为响应式函数编程的iOS实现,ReactiveCocoa将iOS编程带入了一个新纪元,其最大的贡献在于:

统一消息传递机制:iOS 开发中有着各种消息传递机制,包括 KVO、Notification、delegation、block 以及 target-action 方式。

ReactiveCocoa的入门可以参考
http://www.raywenderlich.com/62699/reactivecocoa-tutorial-pt1
http://www.raywenderlich.com/62796/reactivecocoa-tutorial-pt2

这里我着重要说的是ReactiveCocoa (订阅)subscription的实现原理--block回调的应用。在ReactiveCocoa里,RACSignal, RACSubscriber, RACDisposal 是最核心的三个类。下面我们就一个简单的例子深入理解subscription的实现原理,见下面代码:

-(RACSignal *)signInSignal {
  // part 1:[RACSignal createSignal]来获得signal
  return [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
           [self.signInService signInWithUsername:self.usernameTextField.text
                                         password:self.passwordTextField.text
                                         complete:^(BOOL success) {
  // part 3: 进入didSubscribe,通过[subscriber sendNext:]来执行next block
             [subscriber sendNext:@(success)];
             [subscriber sendCompleted];
            }];
            return nil;
          }];
}

// part 2 : [signal subscribeNext:]来获得subscriber,然后进行subscription
[[self signInSignal] subscribeNext:^(id x) { 
    NSLog(@"Sign in result: %@", x); 
}];
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,255评论 4 61
  • Swift版本点击这里欢迎加入QQ群交流: 594119878最新更新日期:18-09-17 About A cu...
    ylgwhyh阅读 25,573评论 7 249
  • 对保龄球的印象,当然就是电视屏幕里的那些。差不多,和高尔夫球是一个级别的运动项目,对我来说。昨天下班后公司活动是打...
    一只默阅读 621评论 0 0
  • 我就是故事里的老男孩 事实上,这些年来我在网络写过了不少随笔,杂文,小说。我唯一记得清楚的是,我从来没有一次以自己...
    周适鲁阅读 366评论 7 21
  • 周末睡个大懒觉,起来做点想吃的菜,然后窝在沙发看电视吃零食。就这样当一个废物的感觉,真棒!
    Candy果阅读 203评论 5 2