0%

block 备忘录

前言

随著 blockiOS4.0OS X 10.6的引入,给事件传递一种新的方式实现,在开发中用得最多的场景莫过于事件回调。使用 block 相对与 delegate 的优势在于,业务集中,可读性强,代码内联,不像代理需要实现很多函数,在适当的场景选择这种方式实现事件传递或者传参效果非常好,现在很多开源项目都实现了两种方法的事件回调。

block 用起来虽然很爽,但也有它的不足,存在循环引用,轻者内存泄露,甚至导致App崩溃,不易调试追溯,因此使用它使一定要小心。鉴于实践中的踩过各种坑,总结下来,方便自己和他人以后查阅,这就是block备忘录写作的初衷。

block 的本质

block 实际上是指向结构体的指针,编译时, block 的内部代码生产对应的函数。

具体结构如下:

block 结构图

与 C 语言的函数指针的区别

  • block 的代码是内联的,效率高于函数调用
  • block 对于外部变量默认是只读属性
  • blockObjective-C 看成是对象处理

block 声明

  • 作为 property > @property (nonatomic, copy) returnType (^blockName)(parameterTypes);

  • 作为方法参数

    - (void)someMethodThatTakesABlock:(returnType (^)(parameterTypes))blockName;

  • 作为一个方法调用参数

    [someObject someMethodThatTakesABlock:^returnType (parameters) {…}];

  • 作为一个 typedef > typedef returnType (^TypeName)(parameterTypes);

    TypeName blockName = ^returnType(parameters) {…};

  • 作为函数参数

    int (^sumOfNumbers)(int a, int b) = ^(int a, int b) {
    return a + b;
    };

SDWebImage 中使用的 block 示例:

1
2
3
4
5
typedef void(^SDWebImageCompletionBlock)(UIImage *image, NSError *error, SDImageCacheType cacheType, NSURL *imageURL);

typedef void(^SDWebImageCompletionWithFinishedBlock)(UIImage *image, NSError *error, SDImageCacheType cacheType, BOOL finished, NSURL *imageURL);

typedef NSString *(^SDWebImageCacheKeyFilterBlock)(NSURL *url);

block 调用

跟 C 函数类似使用 () ,括号里面还可以带一个或者多个参数

1
2
3
4
5
6
7
8
9
10
// block 声明
void)(^loggerBlock)(void);

// block 定义
loggerBlock = ^{
NSLog("hello world")
};

// block 调用
loggerBlock();

block 内存管理

默认情况下, block 是在栈内存中,它不会对所引用的对象进行任何操作;如果对 block 进行一次 copy 操作, block 就会在堆内存中,并且它会它所有的引用的对象做一次 retain 操作

  1. 对于 block 外的变量引用,block 默认是将其复制到其数据结构中来实现访问的
  2. 对于用 __block 修饰的外部变量引用,block 是复制其引用地址来实现访问的
  3. 而 block 会捕获代码外的局部变量,并且仅限于只读操作
  4. 在 block 中希望修改的外界局部对象,必须加上 __block 关键词

ARC

如果对象使用 `__unsafe_unretained` 或 `__weak` 修饰,就不会对其做 `retain` 操作

MRC

如果对象使用了 `__block` 修饰, 就不会对其做 `retain` 操作

为了防止 block 中的循环引用,可以用 __weak 关键词把相应的对象声明为弱引用,在 block 快内部需要多次访问,防止该对象被释放,可以用 __strong 关键词将声明为强引用:

1
2
3
4
5
6
7
8
9
__weak __typeof__(self) weakSelf = self;

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

__strong __typeof(self) strongSelf = weakSelf;

[strongSelf doSomething];
[strongSelf doOtherThing];
});

参考链接