ios定时器

实现方式

1.NSTimer
2.CADisplayLink
3.GCD定时器

一、NSTimer

创建方式
NSTimer *timer1 = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(action) userInfo:NULL repeats:YES];
NSTimer *timer2 = [NSTimer timerWithTimeInterval:1 target:self selector:@selector(action) userInfo:NULL repeats:YES];

第一种已经自动将定时器加入了RunLoop中,模式默认是NSDefaultRunLoopMode
第二种需要将创建的定时器需要手动加入RunLoop中:[[NSRunLoop mainRunLoop] addTimer:timer2 forMode:NSDefaultRunLoopMode]

需要注意的是
1.UIScrollView 滑动时执行的是 UITrackingRunLoopMode,NSDefaultRunLoopMode被挂起,会导致定时器失效,等恢复为滑动结束时才恢复定时器。

要想在scrollview滑动的时候不影响计时

[[NSRunLoop mainRunLoop] addTimer:timer forMode:NSRunLoopCommonModes]
或者
[[NSRunLoop mainRunLoop] addTimer:timer forMode: UITrackingRunLoopMode]
[[NSRunLoop mainRunLoop] addTimer:timer forMode: NSDefaultRunLoopMode]

2.注意相互应用的问题

//timer是self的属性,self持有timer    而timer有通过target:self持有了self形成了死锁,所以通过下面的释放是不行的。
-(void)dealloc{
    [_timer invalidate];
    _timer = nil;
}
应该(这是灵活的,也可以在点击返回按钮的时候销毁掉timer)
-(void)viewWillDisappear:(BOOL)animated{
    [_timer invalidate];
    _timer = nil;
}
特点

NSTimer 需要加入某个RunLoop中,所以NSTimer的准确性跟加入的RunLoop和RunLoopMode有关,如果此RunLoop正在执行一个连续性的运算,timer就会被延时出发。

二、CADisplayLink

CADisplayLink对象是一个和屏幕刷新率同步的定时器对象。屏幕每刷新一次就执行一次回调方法,适合做界面渲染
正常情况iOS设备的屏幕刷新频率是固定60Hz,如果CPU过于繁忙,无法保证屏幕60次/秒的刷新率,就会导致跳过若干次调用回调方法的机会,跳过次数取决CPU的忙碌程度。

//创建
self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(action)];
//间隔的刷新次数
self.displayLink.frameInterval = 60;
//添加RunLoopMode这里和NSTimer一样
[self.displayLink addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSDefaultRunLoopMode];
//释放
[self.displayLink invalidate];
self.displayLink = nil;

二、GCD定时器

GCD精度非常高

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), 1*NSEC_PER_SEC, 0);//每秒执行  第二个参数,当我们使用 dispatch_time 或者 DISPATCH_TIME_NOW 时,系统会使用默认时钟来进行计时。然而当系统休眠的时候,默认时钟是不走的,也就会导致计时器停止。使用 dispatch_walltime 可以让计时器按照真实时间间隔进行计时.
//第三个参数表示间隔时间。
    
dispatch_source_set_event_handler(timer, ^{
      NSLog(@"bb===bb");
      dispatch_async(dispatch_get_main_queue(), ^{
            //这里返回主线程去跟新界面
      });
      if (2==3) {
            //  一定要有dispatch_suspend(_timer)或dispatch_source_cancel(_timer)这两句话来指定出口, 否则定时器不执行 若是要无限循环 就像我这样写让其永远执行不到。
            dispatch_source_cancel(timer); //关闭定时器  
      }
        
});
//开启定时器
dispatch_resume(timer);

GCD计时器开启后立刻执行,而NSTimer是经过一个间隔时间后才执行第一次。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容