![](https://img.haomeiwen.com/i2149459/d6de1930de34efef.jpg)
__block可以用于解决block内部无法修改auto变量值的问题
__block不能修饰全局变量、静态变量(static)
编译器会将__block变量包装成一个对象
__block int age = 18;
^{
NSLog(@"%d", age);
}();
struct __main_block_impl_0 {
struct __block_impl impl;
struct __main_block_desc_0* Desc;
__Block_byref_age_0 *age;// by ref
};
struct __Block_byref_age_0 {
void *__isa;
__Block_byref_age_0 *__forwarding;
int __flags;
int size;
int age;
};
![](https://img.haomeiwen.com/i2149459/a90bb47de998c4cf.png)
-
__block的内存管理
-
当block在栈上时,并不会对__block变量产生强引用
-
当block被copy到堆时
会调用block内部的copy函数
copy函数内部会调用_Block_object_assign函数
_Block_object_assign函数会对__block变量形成强引用(retain)
-
当block从堆中移除时
会调用block内部的dispose函数
dispose函数内部会调用_Block_object_dispose函数
_Block_object_dispose函数会自动释放引用的__block变量(release)
-
__block的__forwarding指针
![](https://img.haomeiwen.com/i2149459/e84d6d6d95787643.png)
![](https://img.haomeiwen.com/i2149459/435d01a07299b447.png)
-
对象类型的auto变量、__block变量
-
当block在栈上时,对它们都不会产生强引用
-
当block拷贝到堆上时,都会通过copy函数来处理它们
__block变量(假设变量名叫做a)
_Block_object_assign((void*)&dst->a, (void*)src->a, 8/*BLOCK_FIELD_IS_BYREF*/);
对象类型的auto变量(假设变量名叫做p)
_Block_object_assign((void*)&dst->p, (void*)src->p, 3/*BLOCK_FIELD_IS_OBJECT*/);
- 当block从堆上移除时,都会通过dispose函数来释放它们
__block变量(假设变量名叫做a)
_Block_object_dispose((void*)src->a, 8/*BLOCK_FIELD_IS_BYREF*/);
对象类型的auto变量(假设变量名叫做p)
_Block_object_dispose((void*)src->p, 3/*BLOCK_FIELD_IS_OBJECT*/);
![](https://img.haomeiwen.com/i2149459/879ea8e937b67bb7.png)
-
被__block修饰的对象类型
-
当__block变量在栈上时,不会对指向的对象产生强引用
-
当__block变量被copy到堆时
会调用__block变量内部的copy函数
copy函数内部会调用_Block_object_assign函数
_Block_object_assign函数会根据所指向对象的修饰符(__strong、__weak、__unsafe_unretained)做出相应的操作,形成强引用(retain)或者弱引用(注意:这里仅限于ARC时会retain,MRC时不会retain) -
如果__block变量从堆上移除
会调用__block变量内部的dispose函数
dispose函数内部会调用_Block_object_dispose函数
_Block_object_dispose函数会自动释放指向的对象(release) -
循环引用问题
![](https://img.haomeiwen.com/i2149459/00d66004dc9d47d0.png)
- 解决方案(ARC):
用__weak
、__unsafe_unretained
解决
用__block
解决(必须要调用block) - 解决方案(MRC):
用__unsafe_unretained
解决
用__block
解决
网友评论