在ARC与非ARC环境下对block使用不当都会引起循环引用问题,一般表现为,某个类将block作为自己的属性变量,然后该类在block的方法体里面又使用了该类本身,简单说就是self.theBlock = ^(void){[self dosomething];或者self.otherVar = XXX;或者_otherVar = ...};block的这种循环引用会被编译器捕捉到并及时提醒,那要如何避免呢,我在这里做了个demo测试。
直接看demo
#import "ViewController.h"
#define CJSUPARC 1
typedef void(^myBlock)(void);
@interface ViewController ()
@property (nonatomic, copy)myBlock theBlock;
@end
@implementation ViewController
- (void)dealloc
{
#if CJSUPARC
[super dealloc];
#else
#endif
NSLog(@"ViewController dealloc");
}
- (void)viewDidLoad {
[super viewDidLoad];
#if CJSUPARC
__block typeof(self) wSelf = self;
NSLog(@"%@",@(self.retainCount));
myBlock block = ^(void){
NSLog(@"%@",@(wSelf.retainCount));
NSLog(@"%@.block init",[wSelf.class description]);
};
self.theBlock = [[block copy] autorelease];
if (self.theBlock) {
self.theBlock();
}
#else
// __block typeof(self) wSelf = self;
// __unsafe_unretained typeof(self) wSelf = self;
__weak typeof(self) wSelf = self;
myBlock block = ^(void){
NSLog(@"%@.block init",[wSelf.class description]);
};
self.theBlock = [block copy];
if (self.theBlock) {
self.theBlock();
}
#endif
}
@end
MRC模式下运行
2016-01-09 23:50:33.993 CJBlockDemo[52536:2193676] self.retainCount = 6
2016-01-09 23:50:33.994 CJBlockDemo[52536:2193676] self.retainCount = 6
2016-01-09 23:50:33.994 CJBlockDemo[52536:2193676] ViewController.block init
2016-01-09 23:50:40.437 CJBlockDemo[52536:2193676] ViewController dealloc
可以看到使用__block能够避免引起循环引用的问题ARC模式下运行
使用__block
2016-01-09 23:50:33.994 CJBlockDemo[52536:2193676] ViewController.block init
虽然编译器警告是没有了,但ViewController却没有执行dealloc函数,说明在arc中__block还是会引起retain。
使用__unsafe_unretained
使用__weak
2016-01-09 23:50:33.994 CJBlockDemo[52536:2193676] ViewController.block init
2016-01-09 23:50:40.437 CJBlockDemo[52536:2193676] ViewController dealloc
都可以避免循环引用的问题,但由于前者是unsafe的,会造成野指针问题,所以尽量少用unsafe_unretained关键字
另外在多线程环境下(block中的wSelf有可能被析构的情况下),需要先将self转为strong指针,避免在运行到某个关键步骤时self对象被析构。
可参考AFNetworking代码:
__weak __typeof(self)weakSelf = self;
AFNetworkReachabilityStatusBlock callback = ^(AFNetworkReachabilityStatus status) {
__strong __typeof(weakSelf)strongSelf = weakSelf;
strongSelf.networkReachabilityStatus = status;
if (strongSelf.networkReachabilityStatusBlock) {
strongSelf.networkReachabilityStatusBlock(status);
}
};
- 第一行:__weak __typeof(self)weakSelf = self;
为防止callback内部对self强引用,weak一下。
其中用到了__typeof(self),这里涉及几个知识点:
- __typeof、__typeof__、typeof的区别
恩~~他们没有区别,但是这牵扯一段往事,在早期C语言中没有typeof这个关键字,__typeof、__typeof__是在C语言的扩展关键字的时候出现的。
typeof是现代GNU C++的关键字,从Objective-C的根源说,他其实来自于C语言,所以AFNetworking使用了继承自C的关键字。 - 对于老的LLVM编译器上面这句话会编译报错,所以在很早的ARC使用者中流行__typeof(&*self)这种写法,原因如下
大致说法是老LLVM编译器会将__typeof转义为 XXX类名 *const __strong的__strong和前面的__weak关键字对指针的修饰又冲突了,所以加上&*对指针的修饰。
第三行:__strong __typeof(weakSelf)strongSelf = weakSelf;
self转回strong了,这里__typeof()里面写的是weakSelf,里面写self也没有问题,因为typeof是编译时确定变量类型,所以这里写self 不会被循环引用。第四、五、六行,如果不转成strongSelf而使用weakSelf,后面几句话中,有可能在第四句执行之后self的对象可能被析构掉,然后后面的StausBlock没有执行,导致逻辑错误。
最后第五行,使用前对block判空。
原文链接:https://www.jianshu.com/p/a87ed8a4363a =========================看法分割线===============
是否所有的Block中,使用self 都会导致循环引用?
系统自带Block不会发生循环引用
如图,使用系统自带的UIView 的Blcok,控制器能被销毁-->说明没有发送循环引用。
原理: UIView的调用的是类方法,当前控制器不可能强引用一个类 ,所以循环无法形成 --> 动画block不会造成循环引用的原因。
所以通过实践得出第一个结论--> 并不是所有的Block中使用self,都会导致循环引用!
问题二:面试官问:那除了系统自带的方法中的Block,你在其他Block中使用self 会导致循环引用吗? -->可答:AFN框架!
最常用的数据请求框架-- AFNetWorking框架的Block是否会强引用?
AFN的Block是否会导致循环引用测试
如上图所示,在AFN的 block { xxx self.view } 使用self,并不会导致循环引用!
原理:AFN无循环是因为绝大部分情况下,你的网络类对象是不会被当前控制器引用的,这时就不会形成引用环。(查阅资料得知)
小tips:也可能AFN底层有自己做了操作,这里没探究到AFN框架底层,仅知道AFN不会造成循环引用。
那什么情况下会导致循环引用呢? --> 自定义Block
自定义Block中使用self
添加 viewDidLoad 提示框-->每次进入都打印viewDidLoad,可以确定是否离开视图控制器-->如果是,但是没有调用dealloc --> 循环引用
循环引用
这时候,我们发现循环引用发生了!所有我们答道:“我们在实际开发中,使用自定义Block,在Block { xxx }中使用self,导致了循环引用 ”
循环引用导致的原因: 相互强指向
循环引用原因
如何解决-->使用weakSelf,这个解决方法估计没见过一百次的,都不算是真正参加过iOS面试的。
----------------------------- 华丽分割线--------------------------------------
一个大写的excuse me 写脸上,49行都报警告了,而且提示可能发送循环引用,这你都能因为这样导致循环引用??这面试官如果知道这个,应该不会这么友好的放过你吧?
由于现在学iOS的太多了,所有可能面试官如果对于循环引用比较了解的话,并不会因为我们回答了上面两个问题就放过我们~他可能会接着问:那如果是我们自己写的Block,(非系统和AFN),在Block中使用self,是否一定会发生循环引用~
探究四:自定义Block是否一定会发生循环引用?
在其他控制器声明一个强指向的Block
调用Blcok
执行效果
如图:发现oneVC被销毁了,说明,自己定义的Block,里面使用了self,并不一定会发生循环引用!
原理:block --> 强指向了self,但是self,并没有指向Block!-->并没有一个 self.block
或者 成员变量 @property block ,所有Block并没有被强指向-->没有发送循环引用!
-->Tips:循环引用发生的条件就是持有这个block的对象,被block里边加入的对象持有。
逼格出现了!!华丽分割线! 既然系统的Block、AFN、都不会发生循环引用,自定义Block又有这么明显的提示-->实际开发中不会遇到循环引用??
---------------------------------高逼格分割线-----------------------------------------
实际开发中:使用通知(NSNotifation),调用系统自带的Block,在Block中使用self --> 会发生循环引用。
通知的接收方法
现在iOS的通知已经比较好用了,如图第二个方法,我最常用的,特别方便,不需要写@selector(方法)+ 调用,直接写在Block中,就可以实现接收通知之后实现的代码。
twoVC发送通知 --> 给oneVC
oneVC 接收通知
使用通知-发生循环引用
如图!这才是实际开发中-->真正有可能发生循环引用的地方!确实也是在通知的Block,但是这次的循环引用并没有提示,而且也确实发生了 --> 这才是真正告诉面试官:我们做过有实际开发,并且是在真实的开发环境中遇到了-->真正的循环引用!!
解决办法-->weakSelf!
参考链接 http://www.jb51.net/article/108171.htm