在开始我们本文之前 我们先看下边的代码 运行结果
NSMallocBlock 对auto 类型变量会进行强引用 因为 block 内存分配在堆空间 内部对 auto 变量进行强引用 在调用的时候 不至于 auto 对象类型已经被销毁掉
在执行到48行代码时候 Person并没有被销毁 ,说明 block 内部对Person 对象进行了强引用
NSStackBlock 不会对 auto类型变量进行强引用 block 是在栈空间上的 本身 block 随时都会被销毁 怎么可能会去强引用Person对象 只有在 copy 到堆内存时 才会对 auto 对象类型进行强引用
void testBlock() {
Person *p = [[Person alloc] init];
p.name = @"dzb";
NSLog(@"block = %@",[^{
NSLog(@"%@",p.name);
} class]);
控制台输出结果
2019-02-22 13:10:33.013768+0800 Block01[69004:1625951] block = __NSStackBlock__
2019-02-22 13:10:33.014011+0800 Block01[69004:1625951] Person dealloc
}
如果 block 拷贝到堆上时 block 会调用内部的 copy 函数 对 auto 对象的强引用
static struct __testBlock_block_desc_0 {
size_t reserved;
size_t Block_size;
/// 对 auto 对象进行强引用
void (*copy)(struct __testBlock_block_impl_0*, struct __testBlock_block_impl_0*);
/// 销毁对 auto 变量的强引用
void (*dispose)(struct __testBlock_block_impl_0*);
} __testBlock_block_desc_0_DATA = { 0, sizeof(struct __testBlock_block_impl_0), __testBlock_block_copy_0, __testBlock_block_dispose_0};
_Block_object_assign 函数会根据 auto变量修饰符 (__strong __weak __unsafe_unretained) 做出相应的操作 形成强引用(retain) 或者弱引用
static void __testBlock_block_copy_0(struct __testBlock_block_impl_0*dst, struct __testBlock_block_impl_0*src) {_Block_object_assign((void*)&dst->p, (void*)src->p, 3/*BLOCK_FIELD_IS_OBJECT*/);}
当 block 从堆上移除的时候 会调用内部的 __testBlock_block_dispose_0 函数 , _Block_object_dispose会自动释放引用的 auto 变量(release)
static void __testBlock_block_dispose_0(struct __testBlock_block_impl_0*src) {_Block_object_dispose((void*)src->p, 3;}