在当前控制器(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