iOS - 关于NSTimer的循环引用

现象

在当前控制器(ViewController)的view上添加了一个自定义的view(LXFTimerView), LXFTimerView在成功创建出来后添加了定时器NSTimer并加入RunLoop开始工作, 当在当前控制器里将LXFTimerView移除掉后,定时器还在工作,而且LXFTimerView里的dealloc并没有调用

现象

代码

LXFTimerView.m

#import "LXFTimerView.h"

@interface LXFTimerView()
/** 定时器 */
@property(nonatomic, weak) NSTimer *timer;
@end

@implementation LXFTimerView

- (instancetype)initWithFrame:(CGRect)frame {
    if (self = [super initWithFrame:frame]) {
        [self addTimer];
    }
    return self;
}

- (void)dealloc {
    NSLog(@"LXFTimerView - dealloc");
    [self removeTimer];
}

#pragma mark - 定时器方法
/** 添加定时器方法 */
- (void)addTimer {
    // 创建定时器
    if (self.timer) { return; }
    self.timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(log) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
}
/** 移除定时器 */
- (void)removeTimer {
    [self.timer invalidate];
    self.timer = nil;
}
- (void)log {
    NSLog(@"定时器 -- %s", __func__);
}
@end

ViewController.m

#import "ViewController.h"
#import "LXFTimerView.h"

@interface ViewController ()
/** timerView */
@property(nonatomic, weak) LXFTimerView *timerView;
@end

@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    
    LXFTimerView *timerView = [[LXFTimerView alloc] initWithFrame:CGRectMake(0, 0, self.view.bounds.size.width, 200)];
    timerView.backgroundColor = [UIColor orangeColor];
    self.timerView = timerView;
    [self.view addSubview:timerView];   
}
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    [self.timerView removeFromSuperview];
}
@end

引用关系

引用关系

问题就出在LXFTimerView与NSTimer之间,在创建定时器时执行

[NSTimer scheduledTimerWithTimeInterval: target: selector: userInfo: repeats:];

会将LXFTimerView进行强引用,什么?我怎么知道?看下图

NSTimer

翻译:定时器保持着对target的强引用,直到定时器作废 那为什么LXFTimerView中的timer属性要用weak?? 不用着急,下面�即将揭晓~

解决方案

让定时器指着另一个对象,让那个对象来执行LXFTimerView中需要执行的方法。 引用关系如下图所示

LXFWeakTarget

创建一个继承于NSObject的类 LXFWeakTarget,并提供一个创建定时器的方法(苹果官方的方法,对scheduledTimerWithTimeInterval进行转到定义操作【就是command+左键】就可以得到) LXFWeakTarget.h

#import <Foundation/Foundation.h>
@interface LXFWeakTarget : NSObject
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
@end
#import "LXFWeakTarget.h"

@interface LXFWeakTarget()
@property(nonatomic, weak) id target;
@property(nonatomic, assign) SEL selector;
@end

@implementation LXFWeakTarget
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo {
    // 创建当前类的对象
    LXFWeakTarget *object = [[LXFWeakTarget alloc] init];
    object.target = aTarget;
    object.selector = aSelector;

    return [NSTimer scheduledTimerWithTimeInterval:ti target:object selector:@selector(execute:) userInfo:userInfo repeats:yesOrNo];
}
- (void)execute:(id)obj {
    [self.target performSelector:self.selector withObject:obj]; 
}
@end

在LXFTimerView.m中导入LXFWeakTarget的头文件

#import "LXFWeakTarget.h"

将创建定时器的类改为 LXFWeakTarget

self.timer = [LXFWeakTarget scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(log) userInfo:nil repeats:YES];

现在再来执行一下程序

�执行dealloc

最后缕下思路

  • 我们用一个LXFWeakTarget来替LXFTimerView执行一些操作。
  • 当没有被定时器强引用的LXFTimerView从父控件上被移除时,就会执行dealloc方法,LXFTimerView被销毁。
  • 将定时器作废并设为nil,这样定时器对LXFWeakTarget的引用也没有了,LXFWeakTarget也会被销毁。

好,那“为什么LXFTimerView中的timer属性要用weak”这个问题就不用多加解析了吧。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏HT

基于 HTML5 Canvas 的 3D WebGL 机房创建

对于 3D 机房来说,监控已经不是什么难事,不同的人有不同的做法,今天试着用 HT 写了一个基于 HTML5 的机房,发现果然 HT 简单好用。本例是将灯光、雾...

2207
来自专栏ShaoYL

iOS:KVO/KVC 的概述与使用

3578
来自专栏HansBug's Lab

算法模板——线性筛素数

实现功能:如题,筛出1——N内的所有素数 原理:如phile神犇所言,这次的才算是真正意义上的线性筛素数,其精髓在于if (i mod a[j])=0 then...

27412
来自专栏chenssy

【追光者系列】HikariCP源码分析之字节码修改类库Javassist委托实现动态代理

很多人都会问HikariCP为什么那么快?之前的两篇文章【追光者系列】HikariCP源码分析之FastList 和 【追光者系列】HikariCP源码分析之C...

1642
来自专栏mathor

LeetCode208. 实现 Trie (前缀树)

632
来自专栏Android 研究

APK安装流程详解9——PackageParser解析APK(上)

为了让咱们更好的理解谷歌的安卓团队对PackageParser的定位,我们来看下PackageParser的注释

1101
来自专栏岑玉海

hbase源码系列(十二)Get、Scan在服务端是如何处理?

继上一篇讲了Put和Delete之后,这一篇我们讲Get和Scan, 因为我发现这两个操作几乎是一样的过程,就像之前的Put和Delete一样,上一篇我本来只打...

45310
来自专栏battcn

一起学设计模式 - 责任链模式

定义如下:一个请求有多个对象来处理,这些对象形成一条链,根据条件确定具体由谁来处理,如果当前对象不能处理则传递给该链中的下一个对象,直到有对象处理它为止。 责任...

521
来自专栏Java成神之路

Spring_总结_04_高级配置(二)之条件注解@Conditional

在上一节,我们了解到 Profile 为不同环境下使用不同的配置提供了支持,那么Profile到底是如何实现的呢?其实Profile正是通过条件注解来实现的。

653
来自专栏北京马哥教育

SQL函数汇总【精选篇】

1.绝对值 SQL:select abs(-1) value O:select abs(-1) value from dual 2.取整(大) ...

2689

扫码关注云+社区