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/
欢迎转载,转载请注明出处。

浙公网安备 33010602011771号