iOS Block循环引用的理解

当block被定义成属性,copy类型,被他所属的类强引用,这个时候这个block引用的时候需要__weak。在定义block为属性的时候,block用strong和copy来修饰效果是一样的,他都会拷贝出来分配一个新的地址给他,会把block...

当block被定义成属性,copy类型,被他所属的类强引用,这个时候这个block引用的时候需要__weak。

在定义block为属性的时候,block用strong和copy来修饰效果是一样的,他都会拷贝出来分配一个新的地址给他,会把block放到堆区里面,官方建议copy。但是如果使用assgin或者retain,retain他只是把block的计数+1,效果和assgin一样,这个block还是在栈区里面。
在堆区:内存释放由用户自己管理。
在栈区:内存释放由系统管理,block可能是被系统给释放回收了。

所以循环引用是,我们对象A用copy修饰定义了属性Block-B,这个时候A强引用了B【A->B】,然后A对象作为参数或者附属属性使用在C对象中,C引用了A【C->A】,C这个使用调用A的Block,block这个时候引用C这个对象或者其属性【B->C】,
所以 就是 A->B->C->A.... 然后他们之间相互强引用,这个时候C控制器pop了要释放了,因为ABC之间造成了强引用,所以他们一直会释放不掉。

一般都是用__weak 和__block来弱引用block中要调用的对象或者属性信息。

__weak和 __block的区别:

__weak 是ARC下使用,只能修饰对象信息,不能修饰基本类型,修饰对象不会增加引用。
__block 在ARC和MRC下都可以使用,对象和基本类型都可以修饰,修饰的对象可以再block里面重新被赋值__weak不可以,修饰对象会增加引用。

在MRC下使用__Block是可以避免循环引用的,但是在ARC下使用__block的话来修饰对象的话,

在ARC下使用 __block typeof(self)weakSelf = self;
因为block是用过添加引用来访问实例变量的,所以self会被retain一次,block也是一个强引用,会引起循环引用。使用__weak的话,当self释放的话,weakSelf也会置空。

本文标题为:iOS Block循环引用的理解

基础教程推荐