Block 的循环引用
Block
是在栈上生成的,所以一般使用copy
方法把Block
复制到堆上,避免Block
被立刻释放。
Block
会对内部的变量形成强引用,而如果同时该变量又持有这个Block
,就会导致循环引用而无法释放,从而导致内存泄露。
最常见的就是self
持有Block
,而又在Block
内部调用self
的方法或属性,那self
和Block
就会形成循环引用而无法释放。由于我们习惯在dealloc
中释放对象,但是即使在dealloc
中将Block
释放也没用,因为self
的dealloc
根本不会跑进去。比如:
1 self.MyBlock = ^void(){ 2 3 [self doSomething]; 4 };
其实,最简单的解决方法就是在self
的某个非dealloc
方法中将Block
主动释放,并在需要释放self
之前调用这个方法,这样才能有效的解除引用。但是这种方法使用起来比较麻烦,而且很容易忘记调用。
所以我们一般是在Block
中使用弱引用的self
。下面分别介绍ARC
和MRC
中在Block
中使用弱引用self
的方法。
ARC
1 __weak typeof(self) weakSelf = self; 2 3 self.MyBlock = ^void(){ 4 5 __strong typeof(self) strongSelf = weakSelf; 6 7 [strongSelf doSomething]; 8 };
这样做的好处是不必在Block
直接使用self
,这样就不会对self
进行强引用,只要self
需要释放,self
就会自动释放,Block
也会自动释放。在ARC
中,进入Block
前,需要使用__weak
对self
进行弱引用,并在Block
中使用__strong
对weakSelf
进行强引用。
这样做的另一个好处是,在ARC
中使用__weak
之后,如果self
在某个地方被释放了,那weakSelf
也会被自动置为nil
,这样即使在Block
中使用weakSelf
,也不会访问错误。
而在Block
中使用__strong
则是为了避免在使用Block
的过程中self
被释放导致访问出错。
MRC
1 __block typeof(self) blockSelf = self; 2 3 self.MyBlock = ^void(){ 4 5 if (!malloc_zone_from_ptr(blockSelf)) 6 return; 7 8 __strong typeof(self) strongSelf = blockSelf; 9 10 [strongSelf doSomething]; 11 };
其实,MRC
的基本思路和ARC
是一样的。有两处不同:
-
在
MRC
中使用__block
而不是__weak
进行弱引用,因为在ARC
中使用__block
会对该对象进行强引用。 -
在
MRC
的Block
中使用malloc_zone_from_ptr()
方法判断blockSelf
是否已经被释放,因为MRC
不会对已释放的对象自动置为nil
。
可见,无论是MRC
还是ARC
,解决方法都是类似的。虽然Block
的使用增加了简洁性和便利性,但使用Block
的过程中也要时刻注意避免内存泄露。
How Do I Declare A Block in Objective-C? 总结了声明Block
的几种格式,在开发过程中可以参考使用。
作者:coltfoal
出处:http://www.cnblogs.com/coltfoal/
欢迎转载,转载请注明出处。