利用RunTime解决由NSTimer导致的内存泄漏
来源:互联网 发布:java web项目实例视频 编辑:程序博客网 时间:2024/05/18 03:47
NSTimer使用场景
用NSTimer
来实现每隔一定时间执行制定的任务,例如最常见的广告轮播图,使用NSTimer
实现这个功能很简单代码如下:
NSTimer *_timer;
_timer = [NSTimer timerWithTimeInterval:1 target:self selector:@selector(timerEvent) userInfo:nil repeats: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
执行该方法。
实现代码
@property (nonatomic,strong)id timerTarget;
@end
staticconstvoid * TimerKey = @"TimerKey";
staticconstvoid * weakKey = @"weakKey";
@implementationTableViewController
- (void)viewDidLoad {
[super viewDidLoad];
_timerTarget = [NSObject new]; //初始化timerTarge对象
class_addMethod([_timerTarget class],@selector(timerEvent), (IMP)timMethod,"v@:");//动态创建timerEvent方法
NSTimer *_timer;
_timer = [NSTimer timerWithTimeInterval:1 target:_timerTarget selector:@selector(timerEvent) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:_timer forMode:NSRunLoopCommonModes];//创建计时器target对象为_timerTarget
objc_setAssociatedObject(_timerTarget, TimerKey, _timer, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
objc_setAssociatedObject(_timerTarget, weakKey,self, OBJC_ASSOCIATION_ASSIGN);//将self对象与NSTimer对象与_timerTarget对象关联
}
-(void)timMethod(idself,SEL _cmd){
TableViewController *tabview = objc_getAssociatedObject(self, weakKey);
[tabview performSelector:_cmd];
}
-(void)timerEvent{
NSLog(@"%@",NSStringFromClass([self class]));
}
-(void)dealloc{
NSTimer *timer = objc_getAssociatedObject(_timerTarget, TimerKey);
[timer invalidate];
NSLog(@"%@--dealloc",NSStringFromClass([self class]));
}
这样当视图销毁时因为当前视图不被任何对象所持有,所以会走dealloc
方法,然后NSTimer
执行invalidate
也被销毁释放掉了。
说明
objc_setAssociatedObject(_timerTarget, weakKey, self,OBJC_ASSOCIATION_ASSIGN);在把_timerTarget
与self
关联时关联的属性一定要设置为OBJC_ASSOCIATION_ASSIGN
。OBJC_ASSOCIATION_ASSIGN
为弱指针类型,如果设置为强制针,那么self
与_timerTarget
就会发生相互强引用但是内存不能正确释放。
关于使用到的Runtime
在把_timerTarget
与self
关联时关联的属性一定要设置为OBJC_ASSOCIATION_ASSIGN
。OBJC_ASSOCIATION_ASSIGN
为弱指针类型,如果设置为强制针,那么self
与_timerTarget
就会发生相互强引用但是内存不能正确释放。
关于使用到的Runtime
点击打开链接动态的为类添加一个timerEvent
的Objective-C
方法,这个方法是由C
的timMethod
方法来实现的
void timMethod(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
。
- 利用RunTime解决由NSTimer导致的内存泄漏
- 利用RunTime解决由NSTimer导致的内存泄漏
- 利用RunTime解决由NSTimer导致的内存泄漏
- 利用NSProxy解决NSTimer内存泄漏问题
- 利用NSProxy解决NSTimer内存泄漏问题
- 那些由NSTimer 和performSelector 可能引起的内存泄漏
- 解决由Handler 引起的内存泄漏
- 解决由Handler引发的内存泄漏
- cvLoadImage导致内存泄漏的解决
- 解决rxjava导致的内存泄漏
- Linux线程导致的内存泄漏分析及解决
- 一次内存泄漏导致的OOM实例分析和解决
- 解决Handler导致的内存泄漏真实样例
- 导致内存泄漏的原因
- ThreadLocal 导致的内存泄漏
- InputMethodManager导致的内存泄漏
- Unity中由引用导致的内存泄露由弱引用解决
- 解决了一个隐蔽的内存泄漏——pthread_create后没有detach导致内存持续增长
- 任务总结
- AS中Git与GitHub的使用入门
- [又值奥运季] 2016年里约奥运会--8月10日赛事
- Valid Anagram
- Android入门:Activity四种启动模式
- 利用RunTime解决由NSTimer导致的内存泄漏
- 利用python进行数据分析-时间序列2
- 深入 Java 调试体系,第 2 部分: JVMTI 和 Agent 实现
- bzoj1009(kmp+dp+矩阵快速幂优化)必需复习
- n皇后
- C#入门6.7——字符串的分割
- 腾讯业务体系介绍
- 使用Intent启动Activity
- 工厂模式