本篇文章主要介绍了iOS NSTimer循环引用的几种解决办法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
发生场景
在 Controller B 中有一个 NSTimer
@property (strong, nonatomic) NSTimer *timer;
你创建了它,并挂载到 main runloop
self.timer = [NSTimer scheduledTimerWithTimeInterval:1
target:self selector:@selector(timerAction:) userInfo:nil repeats:true];
然后退出 Controller B 的时候,忘记关掉 timer 了
Controller B 将不会释放,B 与 timer 循环引用。因为创建 timer 的时候把 self 直接写进去了。
方法一
既然不能直接传 self,那传 weakSelf 试试
__weak typeof(self) weakSelf = self;
self.timer = [NSTimer scheduledTimerWithTimeInterval:1
target:weakSelf selector:@selector(timerAction:) userInfo:nil repeats:true];
测试结果还是发生了循环引用,B 没有释放,timer 对 weakSelf 这个变量是强引用的,timer -> weakSelf -> B -> timer,三者之间形成循环引用。
方法二
设置一个包装类,包着 Controller B 放进 timer 中,像这样
方法三
NSTimer 已知是会强引用参数 target:self 的了,如果忘记关 timer 的话,传什么进去都会被强引用。干脆实现一个 timer 算了,timer 的功能就是定时调某个方法,NSTimer 的调用时间是不精确的!它挂在 runloop 上受线程切换,上一个事件执行时间的影响。
利用 dispatch_asyn() 定时执行函数。看下面代码。
- (void)loop {
[self doSomething];
......
// 休息 time 秒,再调 loop,实现定时调用
[NSThread sleepForTimeInterval:time];
dispatch_async(self.runQueue, ^{
[weakSelf loop];
});
}
dispatch_async 中调 loop 不会产生递归调用
dispatch_async 是在队列中添加一个任务,由 GCD 去回调 [weakSelf loop]
这办法解决了timer 不能释放,挂在 runloop 不能移除的问题。
利用这方法,我写了个不会发生循环引用的 timer,controller 释放,timer 也自动停止释放,甚至 timer 的 block 里面可以直接写 self,也不会循环引用。github下载地址
方法四
NSTimer 我之前没遇到过循环引用的问题,因为我一直都是配对使用,在 viewWillAppear 开启,在 viewWillDisappear 关闭,不关闭的话那么多 timer 挂载在 runloop 上感觉挺影响性能和流畅性的,就像管理内存一样,申请和释放配对使用,就不会泄露了,谁申请谁释放的原则。但是很大的团队的话,别人可能会写错,造成泄露,可以从技术上,团队编程规范上解决他。
比如定一些规范,Controller 退出一定要成功销毁,不能泄露内存。Block 里不能写 self 等等。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程学习网。
本文标题为:iOS NSTimer循环引用的几种解决办法


- Android studio实现动态背景页面 2023-05-23
- Flutter实现底部和顶部导航栏 2022-08-31
- 最好用的ios数据恢复软件:PhoneRescue for Mac 2023-09-14
- iOS 对当前webView进行截屏的方法 2023-03-01
- SurfaceView播放视频发送弹幕并实现滚动歌词 2023-01-02
- Android实现监听音量的变化 2023-03-30
- Android实现轮询的三种方式 2023-02-17
- Android MaterialButton使用实例详解(告别shape、selector) 2023-06-16
- 详解flutter engine 那些没被释放的东西 2022-12-04
- 作为iOS开发,这道面试题你能答出来,说明你基础很OK! 2023-09-14