利用RunTime解决由NSTimer导致的内存泄漏

NSTimer使用场景

用NSTimer来实现每隔一定时间执行制定的任务,例如最常见的广告轮播图,使用NSTimer实现这个功能很简单代码如下

NSTimer *_timer;    _timer = [NSTimer timerWithTimeInterval:1target:selfselector:@selector(timerEvent) userInfo:nilrepeats:YES];    [[NSRunLoop currentRunLoop] addTimer:_timer forMode:NSRunLoopCommonModes];


但是要记住只要触发了计时器这种操作在不用时一定要把计时器终止掉

[_timer invalidate];

一般我们终止这个操作都需要在这个界面销毁时。但是我们在初始化NSTimer时指定了触发事件为self,所以说self被NSTimer强引用了,而NSTimer对象又被加入了当前的循环中,所以说NSTimer被 Runloop 强引用了,所以导致self不会被释放掉就不会触发dealloc方法

实际上想这样操作

-(void)dealloc{    [_timer invalidate];}

但是由于self对象被持有,所有不会走dealloc,导致虽然已经退出当前界面了,但是计时器还是一致再执行,出现内存泄漏。

解决方法

思路很简单,初始化NSTimer时把触发事件的target替换成一个单独的对象,然后这个对象中NSTimer的SEL方法触发时让这个方法在当前的视图self中实现。

利用RunTime在target对象中动态的创建SEL方法,然后target对象关联当前的视图self,当target对象执行SEL方法时,取出关联对象self,然后让self执行该方法。

实现代码

@interfaceTableViewController()@property(nonatomic,strong)idtimerTarget;@endstaticconstvoid* TimerKey = @"TimerKey";staticconstvoid* weakKey = @"weakKey";@implementationTableViewController- (void)viewDidLoad {    [superviewDidLoad];    _timerTarget = [NSObjectnew];//初始化timerTarge对象class_addMethod([_timerTarget class],@selector(timerEvent), (IMP)timMethod,"v@:");//动态创建timerEvent方法NSTimer *_timer;    _timer = [NSTimer timerWithTimeInterval:1target:_timerTarget selector:@selector(timerEvent) userInfo:nilrepeats:YES];    [[NSRunLoop currentRunLoop] addTimer:_timer forMode:NSRunLoopCommonModes];//创建计时器target对象为_timerTargetobjc_setAssociatedObject(_timerTarget, TimerKey, _timer, OBJC_ASSOCIATION_RETAIN_NONATOMIC);    objc_setAssociatedObject(_timerTarget, weakKey,self, OBJC_ASSOCIATION_ASSIGN);//将self对象与NSTimer对象与_timerTarget对象关联}voidtimMethod(idself,SEL _cmd){    TableViewController *tabview = objc_getAssociatedObject(self, weakKey);    [tabview performSelector:_cmd];}-(void)timerEvent{NSLog(@"%@",NSStringFromClass([selfclass]));}-(void)dealloc{    NSTimer *timer = objc_getAssociatedObject(_timerTarget, TimerKey);    [timer invalidate];NSLog(@"%@--dealloc",NSStringFromClass([selfclass]));}


这样当视图销毁时因为当前视图不被任何对象所持有,所以会走dealloc方法,然后NSTimer执行invalidate也被销毁释放掉了。

说明

objc_setAssociatedObject(_timerTarget, weakKey, self,OBJC_ASSOCIATION_ASSIGN);

1

在把_timerTarget与self关联时关联的属性一定要设置为OBJC_ASSOCIATION_ASSIGN。OBJC_ASSOCIATION_ASSIGN为弱指针类型,如果设置为强制针,那么self与_timerTarget就会发生相互强引用但是内存不能正确释放。

关于使用到的Runtime

class_addMethod([_timerTarget class],@selector(timerEvent),(IMP)timMethod,"v@:");

1

动态的为类添加一个timerEvent的Objective-C方法,这个方法是由C的timMethod方法来实现的

voidtimMethod(id self,SEL _cmd){    TableViewController *tabview = objc_getAssociatedObject(self, weakKey);    [tabview performSelector:_cmd];}

该方法是取到_timerTarget关联的对象,然后让该对象去执行timerEvent方法。

"v@:"是方法的参数,关于参数解释参考Objective-C type encodings

objc_setAssociatedObject(id object,constvoid*key, id value, objc_AssociationPolicy policy)objc_getAssociatedObject(id object,constvoid*key)

这组方法是设置关联对象与获取关联对象key是关联对象的key。

转载至:http://blog.csdn.net/ggghub/article/details/50240225

你可能感兴趣的:(利用RunTime解决由NSTimer导致的内存泄漏)