NSTimer

NSTimer

**

创建NSTimer

**
创建NSTimer的常用方法是:

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)target selector:(SEL)aSelector userInfo:(id)userInfo repeats:(BOOL)repeats

创建NSTimer的不常用方法是

+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)target selector:(SEL)aSelector userInfo:(id)userInfo repeats:(BOOL)repeats
- (instancetype)initWithFireDate:(NSDate *)date interval:(NSTimeInterval)ti target:(id)target selector:(SEL)aSelector userInfo:(id)userInfo repeats:(BOOL)repeats

三者之间的区别是:
这里写图片描述

scheduledTimerWithTimeInterval相比它的小伙伴们不仅仅是创建了NSTimer对象, 还把该对象加入到了当前的runloop中,runloop的模式为默认模式(NSDefaultRunLoopMode)!

NSTimer只有被加入到runloop, 才会生效, 即NSTimer才会被真正执行

所以说, 如果你想使用timerWithTimeInterval或initWithFireDate的话, 需要使用NSRunloop的以下方法将NSTimer加入到runloop中

- (void)addTimer:(NSTimer *)aTimer forMode:(NSString *)mode

这里写图片描述
也就是说:

NSTimer  *timer=[NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(changeTimeAtTimedisplay) userInfo:nil repeats:YES];

NSTimer  *timer=[NSTimer timerWithTimeInterval:10 target:self selector:@selector(changeTimeAtTimedisplay) userInfo:nil repeats:YES];

NSRunLoop *runloop=[NSRunLoop  currentRunLoop];

[runloop addTimer:timer  forMode:NSDefaultRunLoopMode];

是同效的。(initWithFireDate 方法创建的timer同第二种的使用方式一样)

关于 - (void)fire; 方法

其实他并不是真的启动一个定时器,从之前的初始化方法中我们也可以看到,建立的时候,在适当的时间,定时器就会自动启动,也即NSTimer是不准时的。那么,fire方法的作用是什么呢,官方解释是:

You can use this method to fire a repeating timer without interrupting its regular firing schedule. If the timer is non-repeating, it is automatically invalidated after firing, even if its scheduled fire date has not arrived.

大概意思就是fire并不是启动一个定时器,只是提前触发而已。我们来用一个按钮操作fire方法试验一下

在一个Controller中创建一个NSTimer属性=>

    //创建一个定时器,

self.timer= [NSTimer  scheduledTimerWithTimeInterval:10.0  target:self  selector:@selector(timerAction)  userInfo:nil  repeats:YES];

//当然这个定时器会自动启动,只不多过了十秒之后,才触发

timerAction事件里面:

- (void)timerAction {

static  int  a =0;

NSLog(@"定时开始了---- %d",a++);

}

然后单击一个按钮的时候:

- (IBAction)startTime:(id)sender {

//只是简单地调用一下这个方法,看到底功能是什么

[_timer  fire];

NSLog(@"定时fire了");

}

打印结果是:

2017-09-27 12:06:15.020 runloop–02[16073:598629]定时开始了—- 0

2017-09-27 12:06:15.020 runloop–02[16073:598629]定时fire了

2017-09-27 12:06:16.543 runloop–02[16073:598629]定时开始了—- 1

2017-09-27 12:06:26.542 runloop–02[16073:598629]定时开始了—- 2

2017-09-27 12:06:36.543 runloop–02[16073:598629]定时开始了—- 3

2017-09-27 12:06:46.542 runloop–02[16073:598629]定时开始了—- 4

结果解释:

定时器开始执行一次方法(即10秒之后)timerAction 之后,第一次执行a为0;下一次10秒后,a将为1,但是当我们点击按钮,执行了一次fire之后,定时器提前执行了一次timerAction方法,立即将a加1了;而后再一个10秒之后,定时器又按照设定将a加了1,变成2。。。。。

即 fire 方法只是提前出发定时器的执行,但不影响定时器的设定时间。

当我们,改为NO时,即不让它循环触发时,我们此时再单击开始按钮。会猛然发现,a+1了,但当我们再点击开始按钮时,会发现a不再加1。原因是:我们的定时器,被设置成只触发一次,再fire的时候,触发一次,该定时器,就被自动销毁了,以后再fire也不会触发了。

销毁NSTimer

invalidate 方法

Stops the receiver from ever firing again and requests its removal from its run loop

This method is the only way to remove a timer from an NSRunLoop object
将timer从它的runloop钟移除,所以:

如果想要销毁NSTimer, 那么确定, 一定以及肯定要调用invalidate方法。
repeat为YES的timer需要显示得进行invalidate销毁。

invalidate与=nil

不能简单得把_timer置为nil来销毁timer,原因:

  1. 首先, 是创建NSTimer, 加入到runloop后, 除了ViewController之外iOS系统也会强引用NSTimer对象
    这里写图片描述
  2. 当调用invalidate方法时, 移除runloop后, iOS系统会解除对NSTimer对象的强引用, 当ViewController销毁时, ViewController和NSTimer就都可以释放了这里写图片描述
  3. 当将NSTimer对象置nil时, 虽然解除了ViewController对NSTimer的强引用, 但是iOS系统仍然对NSTimer和ViewController存在着强引用关系
    这里所说的iOS系统对ViewController的强引用, 不是指为了实现View显示的强引用, 而是指iOS为了实现NSTimer而对ViewController进行的额外强引用
NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)self));

_timer = [NSTimer scheduledTimerWithTimeInterval:TimerInterval

target:self  selector:@selector(timerSelector:)  userInfo:nil  repeats:TimerRepeats];

NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)self));

[_timer invalidate];

NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)self));

各位请注意, 创建NSTimer和销毁NSTimer后, ViewController(就是这里的self)引用计数的变化

Retain count is 7
Retain count is 8
Retain count is 7

这里写图片描述

综上所述, 销毁NSTimer的正确姿势应该是

[_timer invalidate]; // 真正销毁NSTimer对象的地方

如果将上述销毁NSTimer的代码放到ViewController的dealloc方法里, 你会发现dealloc还是永远不会走的 (此外还有另一种说法:

其实就是timer对viewController进行了强调应用,原因是因为,如果要让timer运行的时候执行viewController下面的timerSelector:,timer需要知道target,并且保存这个target,以便于在以后执行这个代码 [target performSelector:], 这里的target就是指viewController。所以,timer和viewController是相互强调引用的。 但是这样看起来,就形成了retain cycle。为了解除retain cycle,我觉得,在-(void)invalidate;这个方法下,timer之前保存的target被设置为nil,强制断开了引用环。这点和设置timer = nil是差不多的。 但是invalidate还做了另外一个动作,就是解除了runloop对timer的强调引用,使得timer成功停止。

) 所以 timer只要没有销毁,就一直保持着对target也就是vc的强引用,dealloc方法就不会走。

所以我将上述代码放到ViewController的其他生命周期方法里, 例如ViewWillDisappear中

综上所述, 销毁NSTimer的正确姿势应该是

- (void)viewWillDisappear:(BOOL)animated {

[super viewWillDisappear:animated];

[_timer invalidate];

_timer = nil;

}

参考链接:
NSTimer使用
iOS开发 之 不要告诉我你会用NSTimer!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
引用\[1\]: \[runLoop addTimer:myTimer forMode:NSDefaultRunLoopMode\]; //实际上这步是不需要,scheduledTimerWithTimeInterval已经纳入当前线程运行。如果使用timerWithTimeInterval则需要。 引用\[2\]: \[\[NSRunLoop currentRunLoop\] addTimer:_timer forMode:NSDefaultRunLoopMode\]; //_timer = \[NSTimer scheduledTimerWithTimeInterval:1.f target:self selector:@selector(timerAction) userInfo:nil repeats:YES\]; 引用\[3\]: GCD的定时器不受RunLoop中Mode的影响(RunLoop内部也是基于GCD实现的,可以根据源码看到), 比如滚动TableView的时候,GCD的定时器不受影响;且比NSTimer更加准时。 问题: NSTimer的mode是什么意思? 回答: NSTimer的mode是指定定时器在运行时所处的运行循环模式。在使用NSTimer时,可以通过指定mode来控制定时器在哪些运行循环模式下运行。比如在引用\[1\]和引用\[2\]中,都使用了NSDefaultRunLoopMode作为定时器的运行循环模式。这意味着定时器会在默认的运行循环模式下运行。而GCD的定时器则不受RunLoop中Mode的影响,可以在任何运行循环模式下运行,如引用\[3\]所示。 #### 引用[.reference_title] - *1* [iOS多线程的初步研究(四)-- NSTimer](https://blog.csdn.net/lengshengren/article/details/12905635)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [NSTimer 基本使用和注意事项](https://blog.csdn.net/wutengwei007/article/details/82221069)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值