对于设置为重复执行模式的计时器,并且加入了RunLoop之后,它会一直有效,这种NSTimer很容易形成保留环。例如:
NSTimer *timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(handleTimer) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
timer会对target进行retain操作,相当于持有了self,而因为timer被加入了RunLoop中,除非手动调用invalidate,timer不会失效。
如果timer不失效,就会造成一个引用环:
self持有了timer,而timer又retain了它的target,即self:
这样,就会出现self的内存泄漏。
解决方法:
1. 通过手动调用invalidate让timer失效,从而打破循环引用:
可以在timer消失的时候或者Controller viewDidDisappear的时候调用invalidate让timer失效,并把它设置为nil,这样就可以打破循环引用;
2. 自定义一个作为消息传递中间者的Proxy(继承自NSProxy的类),并让这个Proxy弱持有真正的target,再把这个Proxy设为timer的target。Proxy会把消息转发给真正的target,而又因为是弱持有的,所以不出出现循环引用(保留环)的问题。
可以使用YYTextWeakProxy这个开源库,可以很方便地实现一个作为中间消息转发者的Proxy,实例代码如下:这样就不需要担心NSTimer会导致它target的内存泄漏了
@implementation MyView {
NSTimer *_timer;
}
- (void)initTimer {
YYTextWeakProxy *proxy = [YYTextWeakProxy proxyWithTarget:self];
_timer = [NSTimer timerWithTimeInterval:0.1 target:proxy selector:@selector(tick:) userInfo:nil repeats:YES];
}
- (void)tick:(NSTimer *)timer {...}
@end