使用 clang 可以将 Objective-C 编译成 C++ 代码,终端指令 clang -rewrite-objc file.m
>> Block 是什么?
- Block 是将函数及其执行上下文封装起来的 OC 对象
>> Block 的实质是什么?
struct __Block_byref_i_0 {
void *__isa;
__Block_byref_i_0 *__forwarding;
int __flags;
int __size;
int var;
};
- 在内存中以结构体的形式存放,包含 isa、函数指针
>> Block 和 Delegate 区别是什么?
- Block 是一个对象,Delegate 是一种设计模式
- 堆 Block强引用对象,在 ARC 下每一次赋值需要从栈拷贝到堆的操作;Delegate 只是保存了一个弱指针,相当于一个查表动作
- 优先使用 Block,如果回调有多个状态,建议使用 Delegate
- Block 可读性较高,可以直接使用上下文,只需要实现即可,可用于成员变量、属性、参数;Delegate 则需要实现代理、委托、协议三个部分
>> Block 分类
- 全局 __NSGobalBlock__ - 在全局作用域下实现 Block,存放在数据区,没有用到外部变量或者只用全局变量、静态变量
- 栈 __NSStackBlock__ - 出了函数作用域之后会由系统释放
- 堆 __NSMallocBlock__ - 由程序员管理内存
继承链: __NSGobalBlock__ 、__NSStackBlock__、__NSMallocBlock__ : NSBlock : NSObject
>> 对上面三种 Block 进行 Copy 操作会发生什么?
- 全局 __NSGobalBlock__ - 什么都没有发生
- 栈 __NSStackBlock__ - 从栈复制到堆,副本保存在堆区
- 堆 __NSMallocBlock__ - 引用计数加 1
>> Block 的几种形式
无参无返
void(^function)(void) = ^() {};
有参无返
void(^function)(NSString *) = ^(NSString *string) {};
有参有返
NSString *(^function)(NSString *) = ^(NSString *string) { return @""; };
无参有返
NSString *(^function)(void) = ^() { return @""; };
>> Block 的使用场景
- 不捕获外部变量 - 直接访问全局变量、静态全局变量、静态变量
- 截取外部变量 - Block 结构体中将添加成员变量,访问的时候不是外部变量,只是作为值传递
- 修改外部变量 - 使用 __block 修饰
全局变量可以直接访问,静态变量通过获取指针修改,局部变量只能值传递,因为局部变量会随函数执行结束释放,不能通过获取指针修改
>> ARC 下编译器会自动将 Block 复制到堆区
- block 作为函数返回值
- 将 block 赋值给 __strong 指针
- block 作为 Cocoa API 方法中含有 usingBlock 的方法参数时
- block 作为 GCD 方法参数时
>> MRC
- 在 MRC 下 Block 赋值不会进行 Copy 操作,手动调用 Copy 也会复制到堆区
>> 解决循环引用的问题 __weak
- __weak 修饰的对象销毁时,对象会置为 nil,不产生引用计数
>> 为什么 __weak 之后还需要用 __strong 修饰?
__weak typeof(self) weakSelf = self;
person.block = ^{
__strong typeof(weakSelf) myself = weakSelf;
NSLog(@"age is %d", myself->_age);
};
强引用 weakSelf 是为了避免 weakSelf 在 Block 执行过程中被释放,一般存在于多线程开发中,Block 执行完后会释放 weakSelf,也就不会一直处于强引用状态
>>__weak 的原理 见 objc-weak
Runtime 中 __weak 维护一个哈希表,Key 为所指对象的指针,value 为 weak 指针