欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

iOS NSTimer循环引用的办法

程序员文章站 2023-12-19 21:45:40
在当前控制器(viewcontroller)的view上添加了一个自定义的view(lxftimerview), lxftimerview在成功创建出来后添加了定时器nst...

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

iOS NSTimer循环引用的办法 代码

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

引用关系

iOS NSTimer循环引用的办法

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

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

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

iOS NSTimer循环引用的办法

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

解决方案

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

iOS NSTimer循环引用的办法

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

现在再来执行一下程序

iOS NSTimer循环引用的办法 

最后缕下思路

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

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

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

上一篇:

下一篇: