iOS开发:NSTimer造成的内存泄漏问题

对于设置为重复执行模式的计时器,并且加入了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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值