[iOS-Foundation] NSTimer

参考资料

NSTimer
深入理解RunLoop
《编写高质量iOS与OS X代码的52个有效方法》中第52条:别忘了 NSTimer 会保留其目标对象

定时器

定时器是线程通知自己做某事的一种方法。iOS 中的定时器由 NSTimer 实现,通过它可以在一段时间后执行一次或循环执行多次某一对象上的特定方法。NSTimer 对象必须作为定时源加入到线程的 runloop 中才可以工作,runloop 对象会强引用被加入的 NSTimer 对象。如果不想让定时器继续执行任务,则需要将定时器变为失效状态。只执行一次的定时器会在任务执行后自动变成失效状态,而循环执行的定时器则需要通过调用方法[- invalidate]来将定时器对象变为失效状态。runloop 不再引用失效的定时器对象。而且,失效的定时器对象不能再被重新使用。注意,只能在创建 NSTimer 对象的线程中调用[- invalidate]方法,否则,Runloop 可能无法正确移除定时源。

可以通过[- fire]方法让定时任务直接执行,如果定时器只执行一次,那么定时器自动失效。如果定时器是循环执行,那么该方法并不影响定时器的定期执行。

NSTimer 并不能保证定时任务一定会执行,例如在触发任务的时间点,runloop 恰好在执行一个非常耗时的任务,或者 runloop 的模式中并不包括监听定时源,那么定时任务都无法执行。对于循环执行的定时器,如果错过了多个周期的执行时间点,timer 对象并不会弥补这些执行次数,而只是直到下一次的触发时间点再进行尝试。为了提高系统的灵活性,通过设置 NSTimer 对象的tolerance属性,可以让触发任务的时间点有一定的延迟误差值,这样任务可能会比设定的时间晚一点执行,但一定不会早于设置的时间点。该属性的默认值是0,建议的值在间隔时间的10%以内。对于循环执行的定时器,下一次执行任务的时间点会根据原始时间,而不是延迟后的执行时间来增加时间间隔进行设置。

通过下列方法可以创建一个 timer 对象,并加入到当前线程的 runloop 中。一旦将定时器加入到线程的 runloop 中,定时器就会在设置的时间点执行任务。

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti invocation:(NSInvocation *)invocation repeats:(BOOL)yesOrNo;
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;

通过下列方法可以创建一个 NSTimer 对象,之后可通过 NSRunloop 对象的[- addTimer:forMode:]方法将 timer 加入到指定线程的 runloop 中。

// 创建一个 timer 对象,
+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti invocation:(NSInvocation *)invocation repeats:(BOOL)yesOrNo;
+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
- (instancetype)initWithFireDate:(NSDate *)date interval:(NSTimeInterval)ti target:(id)t selector:(SEL)s userInfo:(nullable id)ui repeats:(BOOL)rep;

NSTimer 涉及的循环引用问题

由于计时器会保留其目标对象,等到自身失效时再释放此对象,所以设置成重复执行模式的计时器,很容易出现循环引用的问题。如下列代码:

#import "TimerTestClass.h"

@interface TimerTestClass ()

@property (nonatomic) NSTimer *timer;

@end

@implementation TimerTestClass

#pragma mark - Override

- (void)dealloc {
    [self.timer invalidate];
}

#pragma mark - Public

- (void)start {
    self.timer = [NSTimer scheduledTimerWithTimeInterval:5 target:self selector:@selector(doSomething) userInfo:nil repeats:YES];
}

- (void)stop {
    [self.timer invalidate];
    self.timer = nil;
}

#pragma mark - Private

- (void)doSomething {
    // Do something
}

@end

如果创建了本类的实例,并调用了[- start]方法,那么实例通过属性保留了计时器,而计时器的目标对象又是实例本身,所以计时器也保留了目标对象,此时就产生了保留环。

当指向 TimerTestClass 类实例的所有外部引用都移走后,因为保留环,该实例仍然会继续存活,且无法回收,于是就造成了内存泄漏。这种内存泄漏问题尤为严重,因为计时器还将继续反复地执行轮询任务,造成更多不必要的资源消耗。

代码里想在系统回收本类实例的过程中令计时器无效,从而打破保留环的方法实际上是不可能的,因为保留环存在的情况下,实例的保留计数并不为0,系统无法将实例回收。要想打破保留环,只有通过调用[- stop]方法,令计时器失效,从而不再引用本类的实例。但无法确保外界对象会在释放最后一个指向本实例的引用之前,一定会调用[- stop]方法,所以这种方式也是不安全的。

这个问题可通过 block 来解决,为 NSTimer 添加分类如下:

#import "NSTimer+BlockSupport.h"

@implementation NSTimer (BlockSupport)

+ (NSTimer *)demo_scheduledTimerWithTimeInterval:(NSTimeInterval)ti block:(void (^)())block repeats:(BOOL)yesOrNo {
    return [self scheduledTimerWithTimeInterval:ti target:self selector:@selector(blockInvoke:) userInfo:[block copy] repeats:yesOrNo];
}

+ (void)blockInvoke:(NSTimer *)timer {
    void (^block) () = timer.userInfo;
    if (block) { block(); }
}

@end

这段代码将计时器所应执行的任务封装成 block,在调用计时器函数时,把它作为 userInfo 参数传入,只要计时器还有效,就会一直保留着它。计时器现在的 target 是 NSTimer 类对象,这是个单例,因此计时器是否会保留它,其实都无所谓。单纯的将计时器任务封装成块还不能解决问题,修改[- start]方法如下:

- (void)start {
    self.timer = [NSTimer demo_scheduledTimerWithTimeInterval:5 block:^{
        [self doSomething];
    } repeats:YES];
}

因为块捕获了 self 变量,所以块要保留实例。而计时器又通过 userInfo 参数保留了块。最后,实例本身还要保留计时器。这时,循环引用的问题依然存在。不过,只要改用弱引用,即可打破保留环:

- (void)start {
    __weak TimerTestClass *weakSelf = self;
    
    self.timer = [NSTimer demo_scheduledTimerWithTimeInterval:5 block:^{
        TimerTestClass *strongSelf = weakSelf;
        [strongSelf doSomething];
    } repeats:YES];
}

因为块捕获的是实例的弱引用,所以 self 不会为计时器所保留。当块开始执行时,立刻生成 strong 引用,以保证实例在执行期间持续存活。这样,当外界指向类实例的最后一个引用将其释放,则该实例就可为系统所回收了,回收过程中还会调用计时器的[- invalidate]方法。

值得注意的是,从 iOS10 开始,NSTimer 类本身就已经支持了通过 Block 传入任务的方式:

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;
+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;
- (instancetype)initWithFireDate:(NSDate *)date interval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;

Runloop

RunLoop 是线程中的事件处理的循环。使用 RunLoop 的目的是让线程在有任务时处理任务,没有任务时处于休眠状态。

RunLoop 接收输入事件来自两种不同的来源:输入源(input source)和定时源(timer source)。输入源传递异步事件,通常消息来自于其他线程或程序。定时源则传递同步事件,发生在特定时间或者重复的时间间隔。两种源都使用程序的某一特定的处理例程来处理到达的事件。

runloop.png

正常情况下,一个线程启动后,开始执行一个任务,当任务执行完便退出线程。如果要让线程可以一直处于监听状态,随时响应事件,就需要在线程内执行一个循环,直到收到退出的信号时,才结束循环。这种模型通常称为 Event Loop,在苹果的开发体系中的实现就是 RunLoop。

所以,RunLoop 实际上就是一个对象,这个对象管理了其需要处理的事件和消息,并提供了一个入口函数来执行上面 Event Loop 的逻辑。线程执行了这个函数后,就会一直处于这个函数内部 "接受消息->等待->处理" 的循环中,直到这个循环结束(比如传入 quit 的消息),函数返回。

苹果分别在 Foundation 层和 Core Foundation 层提供了 NSRunLoop 和 CFRunLoopRef 两个API。

RunLoop 和线程是一一对应的,可以通过 NSRunLoop 的+ mainRunLoop+ currentRunLoop获取主线程和当前线程的 Run Loop。线程默认是没有 Run Loop
的,当第一次获取时,会创建 RunLoop。

一个 RunLoop 中可以有多个 Mode,RunLoop 启动时会应用某一个 Mode。Mode 内包含了 Source/Timer/Observer 三类集合,分别对应了事件的响应,固定时间点的响应和 RunLoop 本身变化的响应。如果需要切换 Mode,只能退出 Loop,再重新指定一个 Mode 进入。Mode 主要是为了分隔开不同组的 Source/Timer/Observer,让其互不影响。这里有一个 CommonModes 的概念,可以把 Source/Timer/Observer 这些响应加入到某一个 Mode 中,也可以加入到 CommonModes 中,而一个 Mode 可以标记为是否是 Common 的,当 Run Loop 使用一个 Mode 时,如果这个 Mode 是 Common 的,所有在 CommonModes 里的响应都会加入其中,使用 CommonModes 可以避免将同一响应分别加入不同的 Mode 中。

苹果公开提供的 Mode 有 NSDefaultRunLoopMode 和 UITrackingRunLoopMode,NSDefaultRunLoopMode 是 App 平时所处的状态,UITrackingRunLoopMode 是追踪 ScrollView 滑动时的状态。

通过 RunLoop 实现 AutoReleasePool,苹果在主线程 RunLoop 里注册Observer,进入 RunLoop 时,创建自动释放池,Loop 准备进入休眠时,释放旧的池并创建新池,Loop 退出时,释放自动释放池。

通过 RunLoop 实现事件响应,苹果注册响应系统事件的 source,当一个硬件事件(触摸/锁屏/摇晃等)发生后,首先由 IOKit.framework 生成一个 IOHIDEvent 事件并由 SpringBoard (iOS 的界面)接收。SpringBoard 只接收按键(锁屏/静音等),触摸,加速,接近传感器等几种 Event,随后用 mach port 转发给需要的 App 进程。接着苹果注册的那个 Source 就会触发回调,把 IOHIDEvent 处理并包装成 UIEvent 进行处理或分发,其中包括识别 UIGesture/处理屏幕旋转/发送给 UIWindow 等。当识别了一个手势时,首先会将响应链打断,随后系统将对应的 UIGestureRecognizer 标记为待处理,苹果注册了一个 Observer 监测 Loop 即将进入休眠的事件,这个 Observer 的回调函数会获取所有刚被标记为待处理的 GestureRecognizer,并执行GestureRecognizer的回调。

通过 RunLoop 实现界面更新,当在操作 UI 时,比如改变了 Frame、更新了 UIView/CALayer 的层次时,或者手动调用了 UIView/CALayer 的 setNeedsLayout/setNeedsDisplay方法后,这个 UIView/CALayer 就被标记为待处理,并被提交到一个全局的容器去。苹果注册了一个 Observer 监听即将进入休眠和即将退出 Loop 的事件,回调的函数里会遍历所有待处理的 UIView/CAlayer 以执行实际的绘制和调整,并更新 UI 界面。

NSTimer 就是向 Run Loop 注册 Timer 类的响应,当到达固定的时间点时,Loop 就会唤醒并执行响应。

通过 RunLoop 实现 PerformSelecter,当调用 NSObject 的 performSelecter:afterDelay: 后,实际上其内部会创建一个 Timer 并添加到当前线程的 RunLoop 中。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容