iOS定时器任务

有3种方式:CADisplayLink 、NSTimer、GCD

CADisplayLink

特点:屏幕刷新时触发一次,会重复调用指定的方法。
CADisplayLink是一个能让我们以和屏幕刷新率同步的频率,将特定的内容画到屏幕上的定时器类。通常情况下,按照iOS设备屏幕的刷新率60次/秒。

CADisplayLink以特定模式注册到runloop后,每当屏幕显示内容刷新结束时,runloop就会向CADisplayLink指定的target发送一次指定的selector消息,即调用方法。

  • 创建
self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(handleDisplayLink:)];    
[self.displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
//当把CADisplayLink对象add到runloop中后,selector就能被周期性调用
  • 属性
`frameInterval`:
设置间隔多少帧`(NSInteger)`调用一次`selector`方法。默认是1,即每帧都调用一次。

`duration `:
表示两次屏幕刷新之间的时间间隔`(CFTimeInterval)`。只读属性。
注意:该属性在`target`的`selector`被首次调用以后才会被赋值。
`selector`的调用间隔时间t 计算方式是:`t = duration × frameInterval`。
  • 停止及释放
[self.displayLink invalidate];  
 self.displayLink = nil;
//  CADisplayLink对象就会从runloop中移除,selector调用也随即停止
  • 优缺点
    iOS设备的屏幕刷新频率是固定的,CADisplayLink在正常情况下会在每次刷新结束都被调用,精确度相当高。
    1、但如果调用的定时方法比较耗时,超过了屏幕刷新周期,就会导致跳过若干次回调机会。
    2、如果CPU过于繁忙,无法保证屏幕60次/秒的刷新率,就会导致跳过若干次调用回调机会。跳过次数取决CPU的忙碌程度。

  • 应用场景
    从原理上可以看出,CADisplayLink适合做界面的不停重绘。比如,视频播放的时候需要不停地获取下一帧用于界面渲染。

NSTimer

  • 创建
    计时器一定要加入RunLoop中,并且选好mode才能运行。
NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(dosomething:) userInfo:nil repeats:NO];
// TimeInterval:执行之前的等待时间
// repeats : 是否需要循环

上面的创建方法创建一个计时器,并自动加入到MainRunloopNSDefaultRunLoopMode中,可以直接使用。
方法scheduledTimerWithTimeInterval在哪个线程创建,就会被加入哪个线程的RunLoop中就运行在哪个线程。
自己创建的Timer,加入到哪个线程的RunLoop中就运行在哪个线程。

创建后,target对象的计数器会加1,直到执行完毕,会自动减1,自动释放。

  • 缺点:

1、存在延迟问题
不管是一次性的还是周期性,timer的实际触发时间,都会与所加入的RunLoop、RunLoop Mode有关。

如果所加入的RunLoop正在执行一个连续性的任务,timer就会被延时触发。对于重复性的timer,如果延迟超过了一个周期,则会在延时结束后立刻执行,并按照之前指定的周期继续执行。

2、页面滑动时,定时器不工作

主线程的RunLoop里有两个预置的模式:
kCFRunLoopDefaultMode、UITrackingRunLoopMode都是"Common"属性。
前者是App平时所处的状态,后者是追踪滑动的状态。

当创建一个 Timer 并加到 DefaultMode 时,Timer 会得到重复回调。但如果出现页面的滑动,RunLoop 会将模式切换为TrackingRunLoopMode(为了不影响滑动操作)但这时 Timer 就不会被回调。

解决:
滑动和定时器调用互不影响,将这个 Timer 加入模式为NSRunLoopCommonModes

// 创建
NSTimer *timer = [NSTimer timerWithTimeInterval:1 target:self selector:@selector(timerAction) userInfo:nil repeats:YES];
// 添加到runloop
[[NSRunLoop mainRunLoop] addTimer:timer forMode: NSRunLoopCommonModes];

获取runloop的方法
[NSRunLoop mainRunLoop]这个适用于主线程中
[NSRunLoop currentRunLoop]主线程/子线程 都适用

  • 释放
    如果是定时器任务循环执行的话,就必须手动关闭释放!
[timer invalidate];
timer = nil;
释放Timer的注意点:

不能在dealloc中释放定时器。因为计时器的repeatsYES时,计时器会强持有self,导致dealloc永远不会被调用,这个类就永远无法被释放。

解决:可以在viewDidDisappear中释放,这样当类需要被回收时,就可以正常进入dealloc中了。

暂停Timer

需求:先暂停,然后再某种情况下再次开启。主要是为了防止它在后台运行,暂用CPU。
比如,在页面消失的时候关闭 ,然后等页面再次打开 ,又开启定时器。

//关闭定时器  
[myTimer setFireDate:[NSDate distantFuture]];  

//开启定时器  
[myTimer setFireDate:[NSDate distantPast]]; 

GCD

执行一次

dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, 2.0 * NSEC_PER_SEC); 
dispatch_after(popTime, dispatch_get_main_queue(), ^(void){ 
    //执行事件
});

重复

NSTimeInterval period = 1.0; // 时间间隔
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);
dispatch_source_set_timer(_timer, dispatch_walltime(NULL, 0), period * NSEC_PER_SEC, 0);  
dispatch_source_set_event_handler(_timer, ^{
    //在这里执行事件
});
dispatch_resume(_timer);  // 启动

定时器的调用,放在主线程中最优! 在gcd dispatch_async中执行可能会无效!

循环引用问题

1、CADisplayLink、NSTimer会对target产生强引用,如果target又对它们强引用,就会引发循环引用。
2、runloop 会对 CADisplayLink、NSTimer产生强引用。

解决:使用weakSelf,或NSProxy(代理对象)

  • weakSelf
// 内部使用 WeakSelf, 并在视图消失前, 关闭定时器
__weak __typeof(self) weakSelf = self;
NSTimer * timer = [NSTimer timerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) {
    NSLog(@"timer");
}];
self.timer = timer;
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
  • NSProxy
.h
// 解决循环引用问题
@interface MyProxy : NSProxy
- (instancetype)initWithObjc:(id)objc;
+ (instancetype)proxyWithObjc:(id)objc;


.m
@interface MyProxy()
@property(nonatomic,weak) id objc;
@end

@implementation MyProxy
- (instancetype)initWithObjc:(id)objc{
    self.objc = objc;
    return self;
}
+ (instancetype)proxyWithObjc:(id)objc{
    return [[self alloc] initWithObjc:objc];
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    return [self.objc methodSignatureForSelector:aSelector];
}
- (void)forwardInvocation:(NSInvocation *)invocation {
    if ([self.objc respondsToSelector:invocation.selector]) {
        [invocation invokeWithTarget:self.objc];
    }
}

用代理对象引用target。

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

推荐阅读更多精彩内容

  • 概述 RunLoop作为iOS中一个基础组件和线程有着千丝万缕的关系,同时也是很多常见技术的幕后功臣。尽管在平时多...
    sumrain_cloud阅读 943评论 0 5
  • Runloop是iOS和OSX开发中非常基础的一个概念,从概念开始学习。 RunLoop的概念 -般说,一个线程一...
    小猫仔阅读 985评论 0 1
  • ios 常用的定时器有三种:NSTime,CADisplayLink和GCD。 NsTimer // 参数:Int...
    殿小七阅读 832评论 0 2
  • 这篇文章主要整理一下面试中会问到的一个知识点:几种计时器的知识点(一)NSTimer1.什么是NSTimer?官方...
    329fd8af610c阅读 1,363评论 0 0
  • 20岁之前,比拼同质化教育20岁之后,比拼差异化学习 阿水是我的一个好朋友。我们有相同的境遇,相同的状态和相同的话...
    大伟传说阅读 149评论 2 1