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

NSTimer

程序员文章站 2024-03-24 16:14:46
...

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

三者之间的区别是:
NSTimer

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

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

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

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

NSTimer
也就是说:

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对象
    NSTimer
  2. 当调用invalidate方法时, 移除runloop后, iOS系统会解除对NSTimer对象的强引用, 当ViewController销毁时, ViewController和NSTimer就都可以释放了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

综上所述, 销毁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!