贝锐智能攀枝花建站部专注攀枝花网站设计 攀枝花网站制作 攀枝花网站建设
成都网站建设公司服务热线:400-028-6601

网站建设知识

十年网站开发经验 + 多家企业客户 + 靠谱的建站团队

量身定制 + 运营维护+专业推广+无忧售后,网站问题一站解决

iOS中NSTimer循环引用的示例分析

这篇文章将为大家详细讲解有关iOS中NSTimer循环引用的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。

创新互联服务项目包括达川网站建设、达川网站制作、达川网页制作以及达川网络营销策划等。多年来,我们专注于互联网行业,利用自身积累的技术优势、行业经验、深度合作伙伴关系等,向广大中小型企业、政府机构等提供互联网行业的解决方案,达川网站推广取得了明显的社会效益与经济效益。目前,我们服务的客户以成都为中心已经辐射到达川省份的部分城市,未来相信会继续扩大服务区域并继续获得客户的支持与信任!

在当前控制器(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 *)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 
@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”这个问题就不用多加解析了吧。

关于“iOS中NSTimer循环引用的示例分析”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,使各位可以学到更多知识,如果觉得文章不错,请把它分享出去让更多的人看到。


当前标题:iOS中NSTimer循环引用的示例分析
链接分享:http://mswzjz.cn/article/pcipjs.html

其他资讯