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








发表评论
◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。