在当前控制器(viewcontroller)的view上添加了一个自定义的view(lxftimerview), lxftimerview在成功创建出来后添加了定时器nstimer并加入runloop开始工作, 当在当前控制器里将lxftimerview移除掉后,定时器还在工作,而且lxftimerview里的dealloc并没有调用
代码
lxftimerview.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
#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
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
#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之间,在创建定时器时执行
1
|
[nstimer scheduledtimerwithtimeinterval: target: selector: userinfo: repeats:]; |
会将lxftimerview进行强引用,什么?我怎么知道?看下图
翻译:定时器保持着对target的强引用,直到定时器作废 那为什么lxftimerview中的timer属性要用weak?? 不用着急,下面即将揭晓~
解决方案
让定时器指着另一个对象,让那个对象来执行lxftimerview中需要执行的方法。 引用关系如下图所示
创建一个继承于nsobject的类 lxfweaktarget,并提供一个创建定时器的方法(苹果官方的方法,对scheduledtimerwithtimeinterval进行转到定义操作【就是command+左键】就可以得到) lxfweaktarget.h
1
2
3
4
|
#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 |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
#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的头文件
1
|
#import "lxfweaktarget.h" |
将创建定时器的类改为 lxfweaktarget
self.timer = [lxfweaktarget scheduledtimerwithtimeinterval:1.0 target:self selector:@selector(log) userinfo:nil repeats:yes];
现在再来执行一下程序
最后缕下思路
- 我们用一个lxfweaktarget来替lxftimerview执行一些操作。
- 当没有被定时器强引用的lxftimerview从父控件上被移除时,就会执行dealloc方法,lxftimerview被销毁。
- 将定时器作废并设为nil,这样定时器对lxfweaktarget的引用也没有了,lxfweaktarget也会被销毁。
好,那“为什么lxftimerview中的timer属性要用weak”这个问题就不用多加解析了吧。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:https://juejin.im/post/5a329003f265da4320034606