iOS 定时器开发详情
目录
概述
NSTimer
performSelector
GCD timer
CADisplayLink
一、概述
在平时的开发任务中,定时器是我们常用的技术。这一节我们来学习iOS怎么使用定时器。
在iOS中用一个timer对象来表示一个定时器,这个timer对象必须关联到一个runloop对象才能够正常运行。也就是说,runloop对象是timer对象的拥有者,当定时器的时间到期时由runloop对象给timer发通知,所以runloop对象是持有timer对象的强引用的。如果是一次性的定时器话,当定时器到期时,runloop放弃持有timer对象的引用。但如果是循环timer的话 runloop会一直持有timer的引用直到timer调用invalidate。
timer对象关联到runloop对象时需要指定一个runloop mode,默认为default。当定时器到期时,并且runloop运行的mode与timer所关联的mode相同情况下,runloop才会给timer发通知。
假设timer关联到default mode,当runloop运行在 tracking modes时(滑动的时候),即使timer到期了也是不会被触发的。
正是因为timer需要runloop才起作用所以timer是有误差的:当一次runloop循环时检查timer是否满足触发条件,如果不满足则等待下次循环再检查。timer的误差大概是50~100ms,可满足一般对精度要求不高的需求。
在iOS开发中,我们可以四种技术来开发定时器:
- NSTimer - timer对象的cocoa类
- performSelector - NSObject的函数
- GCD timer - dispatch 内的
- CADisplayLink - 与屏幕刷新频率一致的timer
一般来说,如果只是延迟执行的定时器,我们多会用display_after、NSObject的performSelector函数。
一般的循环定时器可以用NSTimer跟 dispatch_source的timer。
CADisplayLink则多用于动画或者视频开发当中
二、NSTimer
创建一个timer的步骤是:
1.创建一个NSTImer
2.把timer添加到runloop
self.unscheduledTimer = [NSTimer timerWithTimeInterval:
target:self
selector:@selector(targetMethod:)
userInfo:nil
repeats:NO]; // 添加到runloop
// NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
// [runLoop addTimer:self.unscheduledTimer forMode:NSRunLoopCommonModes];
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)( * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
[runLoop addTimer:self.unscheduledTimer forMode:NSRunLoopCommonModes];
});
上面代码中,创建一个3s后执行的timer,但是2s后才把这个timer添加到runloop中,3s后就会触发timer(也就是添加到runloop后1s)。
假设我们4s钟后才把timer添加到runloop中,这时的timer会立马被触发。
整过程是这样的,当一个timer被添加到runloop时,runloop会检查这个timer是否到期如果到期了就会立刻触发这个timer,如果没有过期就等待下次runloop时再检查。
我们也可以用下面函数生成一个定时器:
[NSTimer scheduledTimerWithTimeInterval:5.0
target:self
selector:@selector(targetMethod:)
userInfo:nil
repeats:NO];
scheduledTimerWithTimeInterval函数做两件事:
1.生成一个timer并用传入的参数配置它
2.把生成的timer以default mode添加到当前的runloop的。
三、performSelector
performSelector则比较简单了,可以用于一般的延迟任务:
[self performSelector:@selector(targetMethod:) withObject:nil afterDelay:3];
四、GCD timer
GCD的dispath_after函数很好用,可以用于一般的延迟任务:
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// 三秒钟后执行
});
还有可以利用dispatch_source来创建timer
__block int count = ;
/*
* dispatch_source_t dispatch_source_create( dispatch_source_type_t type, uintptr_t handle, unsigned long mask, dispatch_queue_t queue)
* 第一个参数指明这是一个timer,当然还可以指定其他类型
* 第二个参数是一个系统资源的句柄,比如文件句柄
* 第三个参数为flag
* 第四个参数是handle block被提交到的queue
*/
dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, , , dispatch_get_main_queue());
if (timer) {
// 设置 timer
dispatch_source_set_timer(timer, DISPATCH_TIME_NOW, * NSEC_PER_SEC, );
// 设置 event handler
dispatch_source_set_event_handler(timer, ^{
NSLog(@"timer fire");
if (count++ == ) {
// 取消dispatch source
dispatch_source_cancel(timer);
}
});
// 设置 event handler
dispatch_source_set_cancel_handler(timer, ^{
NSLog(@"time cancel.");
});
// 启动 dispatch source
// 因为create后还需要配置一些行为,所以需要手动resume dispatch source
dispatch_resume(timer);
// 注意:
// 这里需要保存一下timer到类变量,不然timer是局部变量运行到函数尾部时,这个定时器也就没了
self.timer = timer;
}
需要说明的是,通过dispatch_source_create生成并返回的变量,不能是局部变量。因为如果是局部变量的话,在生命周期结束时这个source就被释放了。这一点与NSTimer不一样,因为当NSTimer被添加到runloop时,runloop就会持有NSTimer的强引用。
所以如果我们需要一个定时器,并想让这个定时器生命周期跟我们的业务类的生命周期一致时,可以用dispatch_source的方式创建定时器,然后把这个定时器保存为我们的业务类的类成员变量。
五、CADispalyLink
CADisplayLink是一个特殊的timer对象,特殊的在于这个timer的触发的频率跟屏幕刷新的频率是一致的。也就是说,每当屏幕刷新一次就会调用一次timer的回调函数,当然我们可以使设置frameInterval属性来指明刷新多少帧后才触发一次timer,frameInterval默认为1.
self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(targetMethod:)];
// [self.displayLink setFrameInterval:24] // 24帧回调一次
[self.displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
iOS 定时器开发详情的更多相关文章
- ios新手开发——toast提示和旋转图片加载框
不知不觉自学ios已经四个月了,从OC语法到app开发,过程虽然枯燥无味,但是结果还是挺有成就感的,在此分享我的ios开发之路中的小小心得~废话不多说,先上我们今天要实现的效果图: 有过一点做APP经 ...
- Search Ads 已经在美国区上线 - iOS 移动开发周报(46)
Search Ads 已经在美国区上线 - iOS 移动开发周报(46) 新闻 Search Ads 上线 苹果的 AppStore 搜索广告终于 正式上线了 https://developer.ap ...
- ios程序开发杂记
ios程序开发杂记 一.程序构建 与一般的程序构建无太大区别,都是源文件编译链接这一套,通常是在mac上做交叉编译,也就是利用xcode里带的ios编译工具集去生成arm架构的ios程序(或是x86的 ...
- iOS应用开发最佳实践
<iOS应用开发最佳实践> 基本信息 作者: 王浩 出版社:电子工业出版社 ISBN:9787121207679 上架时间:2013-7-22 出版日期:2013 年8月 开本:16 ...
- iOS定时器-- NSTimer 和CADisplaylink
iOS定时器-- NSTimer 和CADisplaylink 一.iOS中有两种不同的定时器: 1. NSTimer(时间间隔可以任意设定,最小0.1ms)// If seconds is les ...
- 聚合数据 iOS 项目开发实战:条码查询器
记录下,聚合数据 iOS 项目开发实战:条码查询器:视频地址:http://www.jikexueyuan.com/course/324.html 条码查询API:https://www.juhe.c ...
- 移动开发在路上-- IOS移动开发系列 多线程二
最近太忙没太多的时间,忙碌的码农生活空下来一点时间,都会挤出来看一些技术或者咨询的文章,废话不多说,直奔主题. 接着上一次的继续说. 定时器在多线程的使用 NSRunLoop 是线程相关的基础框架的一 ...
- 中文 iOS/Mac 开发博客列表
中文 iOS/Mac 开发博客列表 博客地址 RSS地址 OneV's Den http://onevcat.com/atom.xml 一只魔法师的工坊 http://blog.ibireme.com ...
- Unity iOS混合开发界面切换思路
Unity iOS混合开发界面切换思路 最近有很多博友QQ 私信 或则 留言联系我,请教iOS和Unity界面之前相互切换的问题,源代码就不私下发你们了,界面跳转功能的代码我直接贴到下面好了,顺带说i ...
随机推荐
- Linux 服务器之间文件传输
linux的scp命令: scp就是secure copy的简写,用于在linux下进行远程拷贝文件的命令,和它类似的命令有cp,不过cp只是在本机进行拷贝不能跨服务器. 有时我们需要获得远程服务器上 ...
- Maven里面多环境下的属性过滤(配置)
情景:通常一个项目都为分为开发环境(dev)和测试环境(test)还有正式环境(prod),如果每次一打包都要手动地去更改配置文件,例如数据库连接配置.将会很容易出差错. 解决方案:maven pro ...
- python核心编程2 第十三章 练习
13-3.对类进行定制.写一个类,用来将浮点型值转换为金额. class MoneyFmt(object): def __init__(self, value=0.0, flag='-'): self ...
- 我所用过的nginx的功能
前言 当我们提起集群时,一般所用的插件就是nginx.nginx功能如今越来越完善.第三方模块也多如牛毛,在此,总结一下不牵扯第三方模块所具有的功能. 基本功能 反向代理 负载均衡 HTTP服务器(动 ...
- 第三章 最简单的C程序设计——顺序程序设计
一.数据的表现形式及其运算 1.常量和变量 在计算机高级语言中,数据有两种表现形式:常量和变量. 1.1.常量 在程序运行过程中,其值不能被改变的量称为常量.如:5,6,32,0.111. 数值常量就 ...
- ABAP CDS - Syntax
The syntax of the DDL and of the DCL of the ABAP CDS comprises elements of the general DDL and DCL o ...
- namenode处于安全模式怎么解决?
当我们在hdfs上操作文件的时候,有时候会报错 ,出现namenode in safemode namenode处于安全模式的原因: 1.NameNode发现集群中DataNode丢失达到一定 ...
- How to enable download EXE files from the Sharepoint website
As we all know,many applications have forbidden to upload and download exe files.Because the e ...
- 揭秘css
这是我看到非常好的一本电子教程,可以当参考手册使用,链接
- spring location设置本地路径
<context:property-placeholder location="file:D:/jdbc.properties"/> 直接在路径前加上 file: