NSTimer使用方式
常用的创建方式有
1
+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;
2
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;
方法1在创建一个NSTimer的时候会自动添加到Runloop当中。但是需要注意的是,这个方法会将NSTimer对象加入到NSDefaultRunLoopMode
这个模式下。如果定时器运行同时还有滚动视图,那么还是需要手动添加到NSRunLoopCommonModes
模式下。
方法2只创建一个NSTimer,并不会加入到Runloop循环中, 需要手动添加到Runloop循环中
官方文档:
You must add the new timer to a run loop, usingaddTimer:forMode:
. Then, after ti
seconds have elapsed, the timer fires, sending the message aSelector
to target
. (If the timer is configured to repeat, there is no need to subsequently re-add the timer to the run loop.)
同时也强调了,如果repeats
为YES
,则不需要重复添加到runloop里。
常用方法
- (void)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.
比如我们把TimeInterval:(NSTimeInterval)ti
中的ti设置为1, 那么结果将是在NSTimer对象创建完成后的一秒开始第一次执行selector:(SEL)aSelector
中的aSelector
。而- (void)fire
方法就是相当于移除开始需要等待的1秒,立刻开始执行selector:(SEL)aSelector
中的aSelector
。
- (void)invalidate;
官方文档
- Stops the timer 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. The NSRunLoop
object removes its strong reference to the timer, either just before the invalidate method returns or at some later point.
If it was configured with target and user info objects, the receiver removes its strong references to those objects as well.- Special Considerations
You must send this message from the thread on which the timer was installed. If you send this message from another thread, the input source associated with the timer may not be removed from its run loop, which could prevent the thread from exiting properly.
官方文档写的很清楚:
此方法是停止定时器,并将它从Runloop中移除掉。
如果NSTimer设置了target:(id)aTarget
和(我认为应该是“或”)userInfo:(nullable id)userInfo
,也会删除对target:(id)aTarget
和userInfo:(nullable id)userInfo
的强引用。
在调用该方法的时候,必须在创建NSTimer的线程中,否则可能会造成与定时器关联的输入源可能不会从其运行循环中删除,这可能会阻止线程正常退出。
相关面试题
- 两种创建NSTimer的类方法有什么区别?
答案很明显了,是否会自动添加到Runloop的区别。
- NSTimer在日常使用中有什么需要注意的?
- 1、可能会造成循环引用。原因是因为NSTimer会强引用
target:(id)aTarget
和userInfo:(nullable id)userInfo
。- 2、注意线程间通信。因为在子线程中创建NSTimer,
selector:(SEL)aSelector
方法也在子线程中执行。- 3、
- (void)invalidate
要在创建该NSTimer的线程中调用。
- NSTimer会不会造成循环引用?怎么解决?
不一定。可重复的会造成循环引用,不可重复的不会造成循环引用。
解决方法一:(最完美的方法)
在
- (void)viewWillDisappear:(BOOL)animated
或- (void)viewDidDisappear:(BOOL)animated
中调用- (void)invalidate
。或者根据自己的业务需求确定调用- (void)invalidate
解决方法二:(残缺的方法)
还有一种方法就是,换用
block
的方法。block
的循环引用怎么解决肯定都知道吧。同理。
当然,这样控制器是会可以正常销毁没问题,但是定时器却没有被从Runloop中移除掉。至于定时器没有从Runloop中移除会有什么不良影响,还求大神指点。。。