iOS NSTimer循环引用的办法

时间:2022-04-03 13:55:14

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

 

 

iOS NSTimer循环引用的办法 代码

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

引用关系

 

 
iOS NSTimer循环引用的办法

 

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

?
1
[nstimer scheduledtimerwithtimeinterval: target: selector: userinfo: repeats:];

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

 

 
iOS NSTimer循环引用的办法

 

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

解决方案

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

 

 
iOS NSTimer循环引用的办法

 

创建一个继承于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];

 

现在再来执行一下程序

 

 

iOS NSTimer循环引用的办法

最后缕下思路

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

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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://juejin.im/post/5a329003f265da4320034606