目标:NSTimer 调试
思考:这段代码回导致内存泄漏吗?会循环引用吗?
[NSTimer scheduledTimerWithTimeInterval:1.0f target:self selector:@selector(timerAction) userInfo:nil repeats:YES];
针对上面的问题,简单解释一下:
1.这样创建 timer 会导致内存泄露
2.这个不是循环引用 控制器没有持有 timer
解释:scheduled 被 rouLoop 持有,所以不会释放 runLoop 持有 timer ,timer 持有 控制器(self) 导致无法释放
循环引用会导致内存泄漏,强引用也会导致内存泄漏,timer 就是强引用导致的内存泄漏
结果:此时控制器的dealloc方法是不会执行的,所以当控制返回到首页的时候,控制器是得不到释放的。
优化方式1: 在viewWillDisappear方法中释放timer
1.这个方法可以解决一部分问题,比如根控制器中push到A控制器,A控制器中含有timer,此时返回根控制器,timer是可以停止的。
2.如果在A控制器中再次push到B控制器,从B控制器到A控制器时,timer会停止。
所以,不建议使用此方法。
优化方式2: 重写 didMoveToParentViewController,当控制器移除时,释放定时器
-(void)didMoveToParentViewController:(UIViewController *)parent{
if (nil == parent) {
if (self.timer != nil) {
[self.timer invalidate];
self.timer = nil;
}
}
}
优化方式3: 通过中间值进行转发 target 这里要使用到runtime
_target = [NSObject new];
class_addMethod([_target class], @selector(timerAction),(IMP)fireIMP, "v@:");
self.timer = [NSTimer timerWithTimeInterval:1.0f target:_target selector:@selector(timerAction) userInfo:nil repeats:YES];
[[NSRunLoop mainRunLoop] addTimer:self.timer forMode: NSDefaultRunLoopMode];
// 添加动态方法实现
void fireIMP(id self,SEL _cmd){
NSLog(@"fireIMP");
}
优化方式4: 消息转发机制 继承NSProxy
自定义类,继承自 NSProxy,
1.重写方法签名,设置转发对象
2.重写转发方法
eg:
![*h.png](https://upload-images.jianshu.io/upload_images/2527578-0092e71599680401.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
![*h.png](https://upload-images.jianshu.io/upload_images/2527578-0092e71599680401.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
使用:
self.proxy = [WeekProxy alloc];
self.proxy.target = self;
self.timer = [NSTimer timerWithTimeInterval:1.0f target:self.proxy selector:@selector(timerAction) userInfo:nil repeats:YES];
[[NSRunLoop mainRunLoop] addTimer:self.timer forMode: NSDefaultRunLoopMode];