0%

《Effective Objective-C 2.0》6. 块与大中枢派发

第 6 章 块与大中枢派发

block、块、Block 块语义相同。

block 是一种可在 C、C++ 及 Objective-C 代码中使用的 “词法闭包”(lexical closure),借由此机制,开发者可将代码像对象一样传递,令其在不同环境(context)下运行。还有个关键的地方是,在定义 block 的范围内,它可以访问到其中的全部变量。

GCD 是一种与 block 有关的技术,它提供了对线程的抽象,而这种抽象则基于 “派发队列” (dispatch queue) 。开发者可将块排入队列中,由 GCD 负责处理所有调度事宜。GCD 会根据系统资源情况,适时地创建、复用、摧毁后台线程(background thread),以便处理每个队列。 此外,使用 GCD 还可以方便地完成常见编程任务,比如编写 “只执行一次的线程安全代码” (thread-safe single-code execution),或者根据可用的系统资源来并发执行多个操作。

第 37 条:理解 “块” 这一概念

块的基础知识

简单的 block:

^{
    //Block implementation here
};

block 语法结构:

typedef returnType(^name)(arguments);

定义一个名为 someBlock 的变量:

void (^someBlock)() = ^{
    //Block implementation here
};

❇️ block 的强大之处:在 block 内部可以访问 block 外部变量:

// 将变量声明为 __block 之后才可以在 Block 内部对此变量进行修改
__block int additional = 5;
// 声明Block块
int (^addBlock)(int a, int b) = ^(int a, int b) {
    additional = 10;
    return a + b + additional;
};
// 使用Block块
int add = addBlock(2, 5);
  • 如果 block 所捕获的变量是对象类型,那么就会自动保留它。系统在释放这个块的时候,也会将其一并释放。

  • block 本身可视为对象,它也有引用计数。

  • 如果将 block 定义在 Objective-C 类的实例方法中,那么除了可以访问类的所有实例变童之外,还可以使用 self 变量。block 总能修改实例变量,所以在声明时无须加 _block。不过,如果通过读写操作捕获了实例变量,那么也会自动把 self 变量一并捕获,因为实例变量是与 self 所指代的实例关联在一起的。

    - (void)anInstanceMethod {
      //...
      void (^someBlock)() = ^{
          _anInstanceVariable = @"Something";
          NSLog(@"_anInstanceVariable = %@",_anInstanceVariable);
      };
      //...
    }
  • 在 block 中,直接访问实例变量和通过 self 来访问是等效的。

  • self 也是个对象,因而 block 在捕获它时也会将其保留。如果 self 所指代的那个对象同时也保留了块,那么这种情况通常就会导致引用循环

块的内部结构

block 对象在栈中的结构:

对应的结构体定义:

struct Block_descriptor {
    unsigned long int reserved;
    unsigned long int size;
    void (*copy)(void *dst, void *src);
    void (*dispose)(void *);
};

struct Block_layout {
    void *isa;
    int flags;
    int reserved;
    void (*invoke)(void *, ...);
    struct Block_descriptor *descriptor;
    /* Imported variables. */
};
  • isa 指针:指向表明该 block 类型的类。
  • flags:按 bit 位表示一些 block 的附加信息,比如判断 block 类型、判断 block 引用计数、判断 block 是否需要执行辅助函数等。
  • reserved:保留变量,我的理解是表示 block 内部的变量数。
  • invoke:函数指针,指向具体的 block 实现的函数调用地址。
  • descriptor:block 的附加描述信息,比如保留变量数、block 对象的大小、进行 copydispose 的辅助函数指针。
  • variables:即捕获到的变量,因为 block 有闭包性,所以可以访问 block 外部的局部变量。这些 variables 就是复制到结构体中的外部局部变量或变量的地址。

全局块、栈块及堆块

  • 虽然 block 是对象,但是其所占的内存区域是分配在中的。这就是说,块只在定义它的那个范围内有效。
  • 给 block 对象发送 copy 消息可以把 Block 从复制到。拷贝后的块,可以在定义它的那个范围之外使用。
void (^block)();
if (/** condition */) {
    block = [^{
        NSLog(@"Block A");
    } copy];
}else {
    block = [^{
        NSLog(@"Block B");
    } copy];
}
block();
  • 全局块(global block)不会捕捉任何状态(比如外围的变量等),运行时也无须有状态来参与。块所使用的整个内存区域,在编译期已经完全确定了,因此,全局块可以声明在全局内存里,而不需要在每次用到的时候于栈中创建。另外,全局块的拷贝操作是个空操作,因为全局块决不可能为系统所回收。这种块实际上相当于单例。下面就是个全局块:
void (^block) () = ^{
    NSLog(@"This is a block");
);
  • 全局的静态 block: _NSConcreteGlobalBlock 类型的 block 要么是空 block,要么是不访问任何外部变量的 block。它既不在栈中,也不在堆中,我理解为它可能在内存的全局区。
  • 保存在栈中的 block:_NSConcreteStackBlock 类型的 block 有闭包行为,也就是有访问外部变量,并且该 block 只且只有有一次执行,因为栈中的空间是可重复使用的,所以当栈中的 block 执行一次之后就被清除出栈了,所以无法多次使用。
  • 保存在堆中的 block:_NSConcreteMallocBlock 类型的 block 有闭包行为,并且该 block 需要被多次执行。当需要多次执行时,就会把该 block 从栈中复制到堆中,供以多次执行。

要点

  • Clang 是开发 Mac OS X 及 iOS 程序所用的编译器。
  • block 块 是 C、C++、Objective-C 中的词法闭包。
  • block 块 可接收参数,也可返回值。
  • block 块 可以分配在栈或堆上,也可以是全局的。分配在栈上的 block 块 可拷贝到堆里,这样的话,就和标准的 Objective-C 对象一样,具备引用计数了。

第 38 条:为常用的块类型创建 typedef

代码块便捷写法:typedefBlock

typedef <#returnType#>(^<#name#>)(<#arguments#>);

示例一:

// 定义 Block 块
typedef int(^EOCSomeBlock)(BOOL flag, int value);

// 使用 Block 块
EOCSomeBlock blcok = ^(BOOL flag, int value) {
    // Implementation
};

示例二:

typedef void(^EOCCompletionHandler)(NSData *data, NSError *error);

// 方法使用 Block块 作为参数
- (void)startingWithCompletionHandler:(EOCCompletionHandler)completion;

❇️ 使用 typedef 类型定义还便于重构 block 的类型签名。

// 新增一个参数,用以表示完成任务所花的时间
typedef void(^EOCCompletionHandler)
        (NSData *data, NSTimeInterval duration, NSError *error);

要点

  • typedef 重新定义 block 类型,可以令 block 变量用起来更加简单。
  • 定义新类型时应遵循现有的命名习惯,勿使其名称与别的的类型相冲突。
  • 不妨为同一个 block 签名定义多个类型别名。如果要重构的代码使用了 block 类型的某个别名,那么只需修改相应的 typedef 中的 block 签名即可,无需改动其他 typedef

第 39 条:用 handler 块降低代码分散程度

为用户界面编码时,一种常用的范式就是 “异步执行任务”(perform task asynchronously)。这种范式的好处在于:处理用户界面的显示及触摸操作所用的线程,不会因为要执行 I/O 或网络通信这类耗时的任务而阻塞。这个线程通常称为主线程(main thread)。

异步方法在执行完任务之后,需要以某种手段通知相关代码。实现此功能有很多办法。常用的技巧是设计一个委托协议(参见第 23 条),令关注此事件的对象遵从该协议。对象成为 delegate 之后,就可以在相关事件发生时(例如某个异步任务执行完毕时)得到通知了。

Delegate 模式:

#import <Foundation/Foundation.h>
@class EOCNetworkFetcher;

@protocol EOCNetworkFetcherDelegate <NSObject>
- (void)networkFetcher:(EOCNetworkFetcher *)fetcher
        didFinishWithData:(NSData *)data;
@end

@interface EOCNetworkFetcher : NSObject
@property (nonatomic, weak) id<EOCNetworkFetcherDelegate> delegate;
- (instancetype)initWithURL:(NSURL *)url;
- (void)start;
@end

// 其他类实现 delegate 协议:
- (void)fetchFooData {
    NSURL *url = [[NSURL alloc] initWithString:
                  @"https://www.pinterest.com"];
    EOCNetworkFetcher *fetcher =
        [[EOCNetworkFetcher alloc] initWithURL:url];
    fetcher.delegate = self;
    [fetcher start];
}

- (void)networkFetcher:(EOCNetworkFetcher *)fetcher didFinishWithData:(NSData *)data {
    // deal with data
}

block 模式:

#import <Foundation/Foundation.h>

typedef void(^EOCNetworkFetcherCompletionHandler)(NSData *data);

@interface EOCNetworkFetcher : NSObject
- (instancetype)initWithURL:(NSURL *)url;
- (void)startWithCompletionHandler:
            (EOCNetworkFetcherCompletionHandler)handler;
@end

// 其他类获取数据:
- (void)fetchFooData {
    NSURL *url = [[NSURL alloc] initWithString:
                  @"https://www.pinterest.com"];
    EOCNetworkFetcher *fetcher =
        [[EOCNetworkFetcher alloc] initWithURL:url];
    // 调用 start 方法时直接以内联形式定义 Completion Handler。
    [fetcher startWithCompletionHandler:^(NSData *data) {
        // deal with data
    }];
}

❇️相比于委托模式,block 可以使代码更清晰整洁、API 更紧致、逻辑关联性更强。

委托模式有个缺点:如果类要分别使用多个获取器下载不同数据,那么就得在 delegate 回调方法里根据传入的获取器参数来切换。

而使用 block 来写的好处是:无须保存获取器,也无须在回调方法里切换。每个 completion handler 的业务逻辑都是和相关的获取器对象一起来定义的。

1. 分别用两个处理程序来处理操作失败和操作成功

这种 API 设计风格很好,由于成功和失败的情况要分别处理,所以调用此 API 的代码也就会按照逻辑,把应对成功和失败情况的代码分开来写,这将令代码更易读懂。而且,若有需要,还可以把处理失败情况或成功情况所用的代码省略。

#import <Foundation/Foundation.h>

typedef void(^EOCNetworkFetcherCompletionHandler)(NSData *data);
typedef void(^EOCNetworkFetcherErrorHandler)(NSError *error);

@interface EOCNetworkFetcher : NSObject
- (instancetype)initWithURL:(NSURL *)url;
- (void)startWithCompletionHandler:
            (EOCNetworkFetcherCompletionHandler)handler
                    failureHandler:
            (EOCNetworkFetcherErrorHandler)failure;
@end

// 其他类使用:
- (void)fetchFooData {
    NSURL *url = [[NSURL alloc] initWithString:
                  @"https://www.pinterest.com"];
    EOCNetworkFetcher *fetcher =
        [[EOCNetworkFetcher alloc] initWithURL:url];
    [fetcher startWithCompletionHandler:^(NSData *data) {
        // deal with data

    } failureHandler:^(NSError *error) {
        // deal with error

    }];
}
2. 把处理失败所需代码与处理成功所用代码,都封装到同一个 completion handle 块里

缺点:由于全部逻辑都写在一起,导致块代码冗长复杂。

优点:能把所有业务逻辑都放在一起使其更加灵活。例如,在传入错误信息时,可以把数据也传进来。有时数据正下载到一半,突然网络故障了。在这种情况下,可以把数据及相关的错误都回传给块。这样的话,completion handler 就能据此判断问题并适当处理了,而且还可利用已下载好的这部分数据做些事情。

总体来说,笔者建议使用同一个块来处理成功与失败情况。

#import <Foundation/Foundation.h>

typedef void(^EOCNetworkFetcherCompletionHandler)
                                (NSData *data, NSError *error);

@interface EOCNetworkFetcher : NSObject
- (instancetype)initWithURL:(NSURL *)url;
- (void)startWithCompletionHandler:
            (EOCNetworkFetcherCompletionHandler)handler;
@end

// 其他类使用:
- (void)fetchFooData {
    NSURL *url = [[NSURL alloc] initWithString:
                  @"https://www.pinterest.com"];
    EOCNetworkFetcher *fetcher =
        [[EOCNetworkFetcher alloc] initWithURL:url];
    // 需要在块代码中检测传人的error变量,并且要把所有逻辑代码都放在一处
    [fetcher startWithCompletionHandler:^(NSData *data, NSError *error) {
        if (error) {
            // handle failure
        }else {
            // handle success
        }
    }];
}

基于 handler 来设计 API 还有个原因,就是某些代码必须运行在特定的线程上,比如,Cocoa 与 Cocoa Touch 中的 UI 操作必须在主线程上执行:

// NSNotificationCenter
- (id <NSObject>)addObserverForName:(nullable NSNotificationName)name
                             object:(nullable id)obj
                              queue:(nullable NSOperationQueue *)queue
                         usingBlock:(void (^)(NSNotification *note))block;

要点

  • 在创建对象时,使用内联的 handler 块将相关业务逻辑一并声明。
  • 在有多个实例需要监控时,如果采用委托模式,那么经常需要根据传入的对象来切换,而若改用 handler 块来实现,则可直接将 block 与相关对象放在一起。
  • 设计 API 时如果用到了 handler 块,那么可以增加一个参数,使调用者可通过此参数来决定应该把 block 安排在哪个队列上执行。

第 40 条:用块引用其所属对象时不要出现保留环

使用 block 很容易导致循环引用:

//  EOCNetworkFetcher.h
#import <Foundation/Foundation.h>

typedef void(^EOCNetworkFetcherCompletionHandler)
                                (NSData *data);

@interface EOCNetworkFetcher : NSObject
@property (nonatomic, strong, readonly) NSURL *url;
- (instancetype)initWithURL:(NSURL *)url;
- (void)startWithCompletionHandler:
            (EOCNetworkFetcherCompletionHandler)completion;
@end

//  EOCNetworkFetcher.m
#import "EOCNetworkFetcher.h"

@interface EOCNetworkFetcher ()
@property (nonatomic, strong, readwrite) NSURL *url;
@property (nonatomic, copy)
            EOCNetworkFetcherCompletionHandler completionHandler;
@property (nonatomic, strong) NSData *downloadData;
@end

@implementation EOCNetworkFetcher

- (instancetype)initWithURL:(NSURL *)url {
    if (self = [super init]) {
        _url = url;
    }
    return self;
}

- (void)startWithCompletionHandler:
        (EOCNetworkFetcherCompletionHandler)completion {
    self.completionHandler = completion;
    // 开启网络请求
    // 设置 downloadData 属性
    // 下载完成后,以回调方式执行 Block
    [self p_requestCompleted];
}

// 为了能在下载完成后通过 p_requestCompleted 方法执行调用者所指定的块,
// 需要把 completion handler 保存到实例变量
// ❇️ _networkFetcher → _completionHandler
- (void)p_requestCompleted {
    if (_completionHandler) {
        _completionHandler(_downloadData);
    }
}

@end

// 某个类可能会创建以上网络数据获取器对象,并用其从URL中下载数据:
@implementation EOCDataModel {
    // ❇️ EOCDataModel → _networkFetcher
    EOCNetworkFetcher *_networkFetcher;
    NSData *_fetchedData;
}

- (void)downloadData {
    NSURL *url = [[NSURL alloc] initWithString:@"http://www.example.com"];
    _networkFetcher = [[EOCNetworkFetcher alloc] initWithURL:url];
    [_networkFetcher startWithCompletionHandler:^(NSData *data) {

        NSLog(@"Request URL %@ finished",_networkFetcher.url);
        // 因为 completion handler 块要设置 _fetchedData 实例变量,所以它必须捕获 self 变量,而 self 指向 EOCDataModel 类
        // ❇️ _completionHandler → EOCDataModel
        _fetchedData = data;

        // 💡等 completion handler 块执行完毕后,再打破保留环,以便使获取器对象在handler 块执行期间保持存活状态。
        _networkFetcher = nil;
    }];
}

问题:

  • 在上例中,唯有 completion handler 运行过后,方能解除保留环。若是 completion handler— 直不运行,那么保留环就无法打破,于是内存就会泄漏。
  • 如果 completion handler 块所引用的对象最终又引用了这个块本身,那么就会出现保留环。
- (void)downloadData {
    NSURL *url = [[NSURL alloc] initWithString:
                  @"http://www.example.com"];
    EOCNetworkFetcher *networkFetcher = [[EOCNetworkFetcher alloc] initWithURL:url];
    [networkFetcher startWithCompletionHandler:^(NSData *data) {
        // completionHandler → networkFetcher.url
        // networkFetcher → completionHandler
        NSLog(@"Request URL %@ finished",networkFetcher.url);
        _fetchedData = data;
    }];
}

解决方法:获取器对象之所以要把 completion handler 块保存在属性里面,其唯一目的就是想稍后使用这个块。可是,获取器一旦运行过 completion handler 之后,就没有必要再保留它了。所以,只需将 p_requestCompleted 方法按如下方式修改即可:

- (void)p_requestCompleted {
    if (_completionHandler) {
        _completionHandler(_downloadData);
    }
    self.completionHandler = nil; // ❇️
}

要想清楚块可能会捕获并保留哪些对 象。如果这些对象又直接或间接保留了块,那么就要考虑怎样在适当的时机解除保留环。

要点

  • 如果 block 所捕获的对象直接或间接的保留了 block 本身,那么就得当心循环引用的问题。
  • 一定要找个适当的时机解除循环引用,而不能把责任推给 API 的调用者。

第 41 条:多用派发队列,少用同步锁

使用同步锁实现同步机制:

  1. @synchronized 同步块:

    - (void)synchronizedMethod {
     @synchronized (self) {
         // ...
         // 根据给定对象,自动创建一个锁,并等待块中的代码执行完毕。
         // 滥用 @synchronized (self) 会降低代码效率
    
     }   // 执行到这段代码结尾处,释放锁。
    }
  2. NSRecursiveLock 递归锁

    • 线程能够多次持有该锁, 而不会出现死锁(deadlock) 现象。
    • 在极端情况下,同步块会导致死锁。

GCD:串行并发队列

@implementation HQLBlockObject {
    dispatch_queue_t _syncQueue;
}

    // 自定义并发队列
    // ⚠️注意到,文章中此处作者使用的是全局并发队列,而在 Ray Wenderlich 的GCD系列教程中使用的是自定义并发队列:原因在于:全局队列中还可能有其他任务正在执行,一旦加锁就会阻塞其他任务的正常执行,因此我们开辟一个新的自定义并发队列专门处理这个问题。
    _syncQueue = dispatch_queue_create("com.effectiveobjectivec.syncQueue", DISPATCH_QUEUE_CONCURRENT);

- (NSString *)someString {
    __block NSString *localSomeString;
    dispatch_sync(_syncQueue, ^{
        localSomeString = _someString;
    });
    return localSomeString;
}

- (void)setSomeString:(NSString *)someString {
    dispatch_barrier_async(_syncQueue, ^{
        _someString = someString;
    })
}

把设置操作与获取操作都安排在序列化的队列里执行,这样的话,所有针对属性的访问操作就都同步了。

栅栏块

在队列中,栅栏块必须单独执行,不能与其他块并行。这只对并发队列有意义,因为串行队列中的块总是按顺序逐个来执行的。并发队列如果发现接下来要处理的块是个栅栏块 (barrier block) ,那么就一直要等当前所有并发块都执行完毕,才会单独执行这个栅栏块。待栅栏块执行过后,再按正常方式继续向下处理。

dispatch_barrier_sync(dispatch_queue_t  _Nonnull queue, ^{

})
dispatch_barrier_async(dispatch_queue_t  _Nonnull queue, ^{

})
- (NSString *)someString {
    // 后台执行
    _syncQueue = dispatch_get_global_queue(0, 0);

    __block NSString *localSomeString;
    // 同步后台队列
    dispatch_sync(_syncQueue, ^{
        localSomeString = _someString;
    });
    return localSomeString;
}

- (void)setSomeString:(NSString *)someString {
    // 异步栅栏队列
    dispatch_barrier_async(_syncQueue, ^{
       _someString = someString;
    });
}

要点

  • 派发队列可用来表述同步语义,这种做法要比使用 @synchronized 块或 NSLock 对象更简单。
  • 将同步与异步派发结合起来,可以实现与普通加锁机制一样的同步行为,而这么做却不会阻塞执行异步派发的线程。
  • 使用同步队列及栅栏块,可以令同步行为更加高效。

第 42 条:多用 GCD,少用 performSelector 系列方法

// 接受的参数就是要执行的选择子
- (id)performSelector:(SEL)aSelector;

该方法与直接调用选择子等效:

[self performSelector:@selector(selectorMethod)];
[self selectorMethod];

特点:编译器要等到运行期才能知道执行的选择子。可以在动态绑定之上再次使用动态绑定,因而可以实现出下面这种功能:

SEL selector;
if ( /** condition A */ ) {
    selector = @selector(foo);
}else if ( /** condition B */ ) {
    selector = @selector(bar);
}else {
    selector = @selector(baz);
}
[self performSelector:selector];

不推荐使用 performSelector 方法的原因:

⚠️ ARC 下会引起内存泄漏:编译器并不知道将要执行的选择子是什么,因此,也就不了解其方法签名及返回值,甚至连是否有返回值都不清楚,而由于编译器不知道方法名,所以就没办法运用 ARC 的内存管理规则来判定返回值是不是应该释放。鉴于此,ARC 采用了比较谨慎的做法,就是不添加释放操作。然而这么做可能导致内存泄漏,因为方法在返回对象时可能已经将其保留了。

即使使用静态分析器,也很难侦测到随后的内存泄漏。

⚠️返回值只能是 void 或对象类型,performSelector 方法的返回值类型是 id。如果想返回整数或浮点数等类型的值,那么就需要执行一些复杂的转换操作了,而这种转换很容易出错。

如果返冋值的类型为 C 语言结构体,则不可以使用 performSelector 方法。

⚠️performSelector 方法还有诸多局限性,传入的参数类型必须是对象类型且最多只能接受 2 个参数、具备延后执行的方法无法处理带有 2 个参数的选择子。

下面是几个使用 Block 的替代方案:

延后执行

✅推荐:

double delayInSeconds = 5.0;
dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW,
                                        delayInSeconds *NSEC_PER_SEC);
dispatch_after(popTime, dispatch_get_main_queue(), ^{
    [self doSomething];
});

❎反对:

[self performSelector:@selector(doSomething)
             withObject:nil
             afterDelay:5.0];

主线程执行

✅推荐:

// 同步主线程(waitUntilDone:YES)
dispatch_sync(dispatch_get_main_queue(), ^{
    [self doSomething];
});

// 异步主线程(waitUntilDone:NO)
dispatch_async(dispatch_get_main_queue(), ^{
    [self doSomething];
});

❎反对:

[self performSelectorOnMainThread:@selector(doSomething)
                       withObject:nil
                    waitUntilDone:NO];

要点

  • performSelector 系列方法在内存管理方面容易有疏失。它无法确定将要执行的 selector 具体是什么,因而 ARC 编译器就无法插入适当的内存管理方法。
  • performSelector 系列方法所能处理的 selector 太过局限了,selector 的返回值类型及发送给方法的参数个数都受到限制。
  • 如果想把任务放在另一个线程上执行,那么最好不要用 performSelector 系列方法而是应该把任务封装到 block 里然后调用 GCD 机制的相关方法来实现。

第 43 条:掌握 GCD 及操作队列的使用时机

GCD & NSOperation

  • GCD 是纯 C 的 API,而 NSOperation(操作队列)则是 Objective-C 的对象。

  • 在 GCD 中,任务用 Block 来表示,而 Block 是个轻量级数据结构(参见第 37 条)。与之相反,“操作”(operation) 则是个更为重量级的 Objective-C 对象

  • 用 NSOperationQueue 类的 addOperationWithBlock: 方法搭配 NSBlockOperation 类来使用操作队列,其语法与纯 GCD 方式非常类似。使用 NSOperation 及 NSOperationQueue 的好处如下:

    • 取消某个操作。运行任务之前,可以在 NSOperation 对象上调用 cancel 方法取消任务执行。
    • 指定操作间的依赖关系。
    • 通过键值观测机制(简称 KVO)监控 NSOperation 对象的属性。
    • 指定操作的优先级。操作的优先级表示此操作与队列中其他操作之间的优先关 系。优先级高的操作先执行,优先级低的后执行。
    • 重用 NSOperation 对象。
  • 操作队列有很多地方胜过派发队列。操作队列提供了多种预设的执行任务的方式,开发者可以直接使用。

  • 有一个 API 选用了操作队列而非派发队列,这就是 NSNotificationCemer,开发者可通过其中的方法来注册监听器,以便在发生相关事件时得到通知,而这个方法接受的参数是块,不是选择子。

    - (id <NSObject>)addObserverForName:(nullable NSNotificationName)name
                               object:(nullable id)obj
                                queue:(nullable NSOperationQueue *)queue
                           usingBlock:(void (^)(NSNotification *note))bloc;
  • 应该尽可能选用高层 API,只在确有必要时才求助于底层。笔者也同意这个说法,但我并不盲从。某些功能确实可以用高层的 Objective-C 方法来做,但这并不等于说它就一定比底层实现方案好。要想确定哪种方案更佳,最好还是测试一下性能。

要点

  • 在解决多线程与任务管理问题时,派发队列并非唯一方案。
  • 操作队列提供了一套高层的 Objective-C API,能实现纯 GCD 所具备的绝大部分功能,而且还能完成一些更为复杂的操作,那些操作若改用 GCD 来实现,则需另外编写代码。

第 44 条:通过 Dispatch Group 机制,根据系统资源状况来执行任务

dispatch group 是 GCD 的一项特性,能够把任务分组。调用者可以等待这组任务执行完毕,也可以在提供回调函数之后继续往下执行,这组任务完成时,调用者会得到通知。这个功能有许多用途,其中最重要、最值得注意的用法,就是把将要并发执行的多个任务合为一组,于是调用者就可以知道这些任务何时才能全部执行完毕。比方说,可以把压缩一系列文件的任务表示成 dispatch group

创建及使用 dispatch group

dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
    // 并行执行的线程一
});
dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
    // 并行执行的线程二
});
dispatch_group_notify(group, dispatch_get_global_queue(0, 0), ^{
   // 汇总结果
});

给任务编组的两种方法:

// 1.比 dispatch_async 多一个参数,用于表示待执行的块所归属的组
dispatch_group_async(dispatch_group_t group,
                     dispatch_queue_t queue,
                     dispatch_block_t block);

// 2.dispatch_group_enter、dispatch_group_leave 需要成对使用
dispatch_group_enter(dispatch_group_t group); // 使分组里正要执行的任务数递增,
dispatch_group_leave(dispatch_group_t group); // 使分组里正要执行的任务数递减.

dispatch_group_wait 函数用于等待 dispatch group 执行完毕:

dispatch_group_wait(dispatch_group_t group, 
                    dispatch_time_t timeout);

示例:

令数组中的每个对象都执行某项任务,并且等待所有任务执行完毕:

dispatch_queue_t queue =
    dispatch_get_global_queue(QOS_CLASS_DEFAULT, 0);
dispatch_group_t group = dispatch_group_create();
for (id object in collection) {
    dispatch_group_async(group, queue, ^{
        [object performTask];
    });
}

dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
// 任务执行完毕后继续操作

// 若当前线程不应阻塞,则可以使用 dispatch_group_notify 代替 dispatch_group_wait
dispatch_queue_t notifyQueue = dispatch_get_main_queue();
dispatch_group_notify(group, notifyQueue, ^{
    // 任务执行完毕后继续操作
});

开发者未必总需要使用 dispatch group。有时候采用单个队列搭配标准的异步派发,也可实现同样效果:

// 自定义串行队列
dispatch_queue_t queue =
    dispatch_queue_create("com.effecitveobjectivec.queue", NULL);
for (id object in collection) {
    dispatch_async(queue, ^{
        [object performTask];
    });
}
dispatch_async(queue, ^{
    // 任务执行完毕后继续操作
});

根据系统资源状况来执行任务:

为了执行队列中的块,GCD 会在适当的时机自动创建新线程或复用旧线程。如果使用并发队列,那么其中有可能会有多个线程,这也就意味着多个块可以并发执行。在并发队列中,执行任务所用的并发线程数量,取决于各种因素,而 GCD 主要是根据系统资源状况来判定这些因素的。假如 CPU 有多个核心,并且队列中有大量任务等待执行,那么 GCD 就可能会给该队列配备多个线程。通过 dispatch group 所提供的这种简便方式,既可以并发执行一系列给定的任务,又能在全部任务结束时得到通知。由于 GCD 有并发队列机制,所以能够根据可用的系统资源状况来并发执行任务。而开发者则可以专注于业务逻辑代码,无须再为了处理并发任务而编写复杂的调度器。

遍历某个 collection ,并在其每个元素上执行任务,而这也可以用另外一个 GCD 函数来实现:

dispatch_apply(size_t iterations,
               dispatch_queue_t queue,
               void (^block)(size_t));

此函数会将块反复执行一定的次数,每次传给块的参数值都会递增,从 0 开始,直至 “iterations-1”。

// 自定义串行队列
dispatch_queue_t queue =
    dispatch_queue_create("com.effecitveobjectivec.queue", NULL);    
dispatch_apply(10, queue, ^(size_t) {
    // Perform Task:0~9
});

与 for 循环不同的是, diSpatch_apply 所用的队列可以是并发队列。但是 diSpatch_apply 会持续阻塞,直到所有任务都执行完毕为止。

假如把块派给了当前队列(或者体系中高于当前队列的某个串行队列),就将导致死锁。若想在后台执行任务,则应使用 dispatch group

要点

  • 一系列任务可归入一个 dispatch group 之中。开发者可以在这组任务执行完毕时获得通知。
  • 通过 dispatch group,可以在并发式派发队列里同时执行多项任务。此时 GCD 会根据系统资源状况来调度这些并发执行的任务。开发者若自己来实现此功能,则需编写大量代码。

第 45 条:使用 dispatch_once 来执行只需运行一次的线程安全代码

单例模式:

+ (instancetype)sharedInstance {
    static EOCClass *sharedInstance = nil;
    // 为保证线程安全,将创建单例实例的代码包裹在同步块里。
    @synchronized (self) {
        if (!sharedInstance) {
            sharedInstance = [[self alloc] init];
        }
    }
    return sharedInstance;
}

dispatch_once 函数:

_dispatch_once(dispatch_once_t *predicate,
               dispatch_block_t block)

该函数保证相关的块必定会执行,且仅执行一次。首次调用该函数时,必然会执行块中的代码,最重要的一点在于,此操作完全是线程安全的

+ (instancetype)sharedInstance {
    static EOCClass *sharedInstance = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        sharedInstance = [[self alloc] init];
    });
    return sharedInstance;
}
  • 使用 dispatch_once 可以简化代码并且彻底保证线程安全。
  • 由于每次调用时都必须使用完全相同的标记,所以标记要声明成 static。把该变量定义在 static 作用域中,可以保证编译器在每次执行 sharedlnstance 方法时都会复用这个变量,而不会创建新变量。
  • 此外,dispatch_once 更高效。 @synchronized 使用了重量级的同步机制,每次运行代码前都要获取锁。而 dispatch_once 采用原子访问(atomic access)来查询标记,以判断其所对应的代码原来是否已经执行过。

要点

  • 经常需要编写只需执行一次的线程安全代码。通过 GCD 所提供的 dispatch_once 函数,很容易就能实现此功能。
  • 标记应该声明在 staticglobal 作用域中,这样的话,在把只需执行一次的 block 传给 dispatch_once 函数时,传进去的标记也是相同的。

第 46 条:不要使用 dispatch_get_current_queue

// 此函数返回当前正在执行代码的队列。
// This function is deprecated and will be removed in a future release.
dispatch_get_current_queue(void);

⚠️ iOS 系统从 6.0 版本起,已经正式弃用此函数了。

要点

  • dispatch_get_current_queue 函数的行为常常与开发者所预期的不同。此函数已废弃,只应做调试之用。
  • 由于派发队列是按层级来组织的,所以无法单用某个队列对象来描述当前队列这一概念。
  • dispatch_get_current_queue 函数用于解决由不可重入的代码所引发的死锁,然而能用此函数解决的问题,通常也能改用 “队列特定数据” 来解决。

参考

  • Objective-C 中的 Block @程序员说
  • Objective-C 中的 Block @onevcat
  • 使用 GCD @唐巧
  • 起底多线程同步锁 (iOS)

欢迎关注我的其它发布渠道