IOS开发定时器延时的探究

在日常开发中,我们经常要延时等待(如网络请求,UI更新完)然后做一些事情,或者是做一些周期性的处理。这个时候就要求我们实现一个周期性的定时器进行延时操作。常用的延时实现方法有下面三种:

1.NStimer实现

不管是一次性的还是周期性的timer的实际触发事件的时间,都会与所加入的RunLoop和RunLoop Mode有关,如果此RunLoop正在执行一个连续性的运算,timer就会被延时触发。重复性的timer遇到这种情况,如果延迟超过了一个周期,则会在延时结束后立刻执行,并按照之前指定的周期继续执行。(PS: 所以有时使用timer,会偶尔出现连续两次刷新的情况,可能上一次timer被阻塞)

NSTimer *timer =[NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(changeUI) userInfo:nil repeats:YES];

[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

上面方式:每隔一秒调用一次定时器的block

使用注意:1. 这个runLoop运行模式,建议使用NSRunLoopCommonModes;使用NSDefaultRunLoopMode,在屏幕上有UI更新(如ScrollView滑动时),runLoop就会被切换到其他的模式(NSEventTrackingRunLoopMode),此时timer不会被执行。NSRunLoopCommonModes中包含了NSEventTrackingRunLoopMode,timer照样被执行

2.当不在使用这个timer时,需要手动调用invalidate方法将timer释放并重置为nil。因为在创建timer时,会默认对target进行一次retain,如果不手动释放timer,target对象也得不到释放会造成内存泄露

3.主线程开启,回调的selector默认也在主线程执行。如果想要多线程执行timer,需要手动开启一个NSthread,值得主要的是:由于子线程的runLoop默认是关闭的,所以需要手动开启子线程的runLoop,否则timer永远不会被执行

//创建并执行新的线程

NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(newThread) object:nil];

[thread start];

//在新线程开启定时器

- (void)openTimer

@autoreleasepool

{

//在当前Run Loop中添加timer,模式是默认的NSDefaultRunLoopMode

[NSTimer scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(timer_callback) userInfo:nil repeats:YES];

//开始执行新线程的Run Loop

[[NSRunLoop currentRunLoop] run];

}

}

2.CADisplayLink:

frameInterval :NSInteger类型的值,用来设置间隔多少帧调用一次selector方法,默认值是1,即每帧都调用一次。

duration:readOnly的CFTimeInterval值,表示两次屏幕刷新之间的时间间隔。需要注意的是,该属性在target的selector被首次调用以后才会被赋值。selector的调用间隔时间计算方式是:调用间隔时间 = duration * frameInterval (调用一次方法刷新帧数 *刷新每一帧所需的时间)

CADisplayLink的使用跟timer比较类似,这里就不详细介绍。朋友们可以参考参考下面这个链接。

http://blog.csdn.net/ch_soft/article/details/9408855

3.GCD实现Timer

GCD中的Timer应该是最灵活的,而且是多线程的。GCD中的Timer是靠Dispatch Source来实现的。

最近使用GCD的timer做一个验证码发送的例子。整个使用过程分为五步,代码如下:

__block NSInteger time = 59; //倒计时时间

//1.创建一个全局并发队列

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

//2.创建一个timer加到队列中

dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);

//3.设置timer的执行周期,最后一个参数表示延时秒数为0,尽可能精确

dispatch_source_set_timer(_timer,dispatch_walltime(NULL, 0),1.0*NSEC_PER_SEC, 0);

//4.设置timer周期执行的block

dispatch_source_set_event_handler(_timer, ^{

if(time <= 0){ //倒计时结束,关闭定时器,还原UI

dispatch_source_cancel(_timer);

dispatch_async(dispatch_get_main_queue(), ^{

//设置按钮的样式

[self.getMsgBtn setTitle:@"获取验证码" forState:UIControlStateNormal];

[self.getMsgBtn setTitleColor:ZGColor(25, 111, 255) forState:UIControlStateNormal];

[self.getMsgBtn setBackgroundColor:[UIColor whiteColor]];

self.getMsgBtn.titleLabel.font = [UIFont systemFontOfSize: 14.0];

self.getMsgBtn.userInteractionEnabled = YES;

});

}else{ //否则继续计时

int leftSeconds = time % 60;

dispatch_async(dispatch_get_main_queue(), ^{

//设置按钮显示读秒效果

[self.getMsgBtn setTitle:[NSString stringWithFormat:@"%d秒后重新发送",leftSeconds] forState:UIControlStateNormal];

[self.getMsgBtn setTitleColor:[UIColor whiteColor] forState:UIControlStateNormal];

[self.getMsgBtn setBackgroundColor:[UIColor lightGrayColor]];

self.getMsgBtn.titleLabel.font = [UIFont systemFontOfSize: 13.0];

self.getMsgBtn.userInteractionEnabled = NO;

});

time--;

}

});

//5.启动timer

dispatch_resume(_timer);

使用注意:由于dispatch_source_t 创建后默认是暂停状态,需要手动开启才会执行。

另外,实际开发中有需求:等待reload执行完刷新完tableView,然后做一些操作

想要程序延迟到reloadData结束在操作,可以用以下方法

方法1:

[self.tableView reloadData];

[self.tableView layoutIfNeeded];

//刷新完成

…….

layoutIfNeeded会强制重绘并等待完成。

方法2:

[objc] view plain copy

[self.tableView reloadData];

dispatch_sync(dispatch_get_main_queue(), ^{

//刷新完成

});

reloadData会在主队列执行,而dispatch_get_main_queue会等待机会,直到主队列空闲才执行。

时间: 2024-07-30 20:30:15

IOS开发定时器延时的探究的相关文章

iOS开发之延时执行

方法1: performSelector(SEL) withObjects:(id) afterDelay:(NSTimeInterval); 方法2: dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ //需要延时执行的代码 });

【iOS开发-14】UIView的属性,父视图和子视图的层级操作,子视图的自适应模式,外加一个定时器

(1)UIView视图frame的设置,四个参数,前2个确定位置,后2个确定大小. (2)UIView的内容模式contentMode,和在UIImage中说的是一样的,而且在UIImage中展示更容易理解. (3)UIView最重要的就是父视图和子视图之间的关系,以及父视图操作子视图(增加一个子视图,把一个子视图放到最下面最上面.交换两个子视图的加载顺序等等) (4)还有一个重要的是,父视图如果发生变化,子视图怎么自动调整布局:先让父视图允许子视图干这个事,即把父视图的属性autoresize

iOS开发,定时器的使用

在iOS开发中,我们可以通过三种途径来实现定时调用某一个方法的功能.为了简便期间,我直接在Xcode中写代码以及注释. 首先我们定义一个被定时执行的方法 - (void)reloop { NSLog(@"循环执行"); } 下边我们分别讨论这三种方法 1.使用NSTimer // 1.使用nstimer创建定时器 // A.自动加入主循环 NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:2 target:self sel

iOS开发——多线程

很多朋友都说iOS开发中,最难理解和学习的就是多线程,很多的原理实现都是通过log看到,也比较抽象,本人也是在多线程方面投入过很多脑细胞..无论这方面的知识掌握和应用起来是否轻松,牢固的基本功.正确的认识理解.再加上充分的实战经验,都能助你将其搞定.这里先介绍一些多线程的知识以及应用,作为讨论,大家共同学习. 一.多线程基本概念 1.线程与进程  (1)进程:操作系统的每一个应用程序就是一个进程 (2)线程:进程的基本执行单元,一个进程的所有任务都在线程中执行 2.主线程 (1)定义:一个程序运

iOS开发工具-网络封包分析工具Charles

简介 本文为InfoQ中文站特供稿件,首发地址为:文章链接.如需转载,请与InfoQ中文站联系. Charles是在Mac下常用的截取网络封包的工具,在做iOS开发时,我们为了调试与服务器端的网络通讯协议,常常需要截取网络封包来分析.Charles通过将自己设置成系统的网络访问代理服务器,使得所有的网络访问请求都通过它来完成,从而实现了网络封包的截取和分析. Charles是收费软件,可以免费试用30天.试用期过后,未付费的用户仍然可以继续使用,但是每次使用时间不能超过30分钟,并且启动时将会有

iOS开发RunLoop学习:四:RunLoop的应用和RunLoop的面试题

一:RunLoop的应用 #import "ViewController.h" @interface ViewController () /** 注释 */ @property (nonatomic, strong) NSThread *thread; @end @implementation ViewController /** * 1:用NSThread创建线程的时候,不要忘记调用start方法来开启线程,在一条线程中的任务执行的顺序是同步的,串行执行,并且当线程中的任务执行完毕后

iOS开发系列--音频播放、录音、视频播放、拍照、视频录制

iOS开发系列--音频播放.录音.视频播放.拍照.视频录制 转载:http://www.cnblogs.com/kenshincui/p/4186022.html#avFoundationCamera --iOS多媒体 概览 随着移动互联网的发展,如今的手机早已不是打电话.发短信那么简单了,播放音乐.视频.录音.拍照等都是很常用的功能.在iOS中对于多媒体的支持是非常强大的,无论是音视频播放.录制,还是对麦克风.摄像头的操作都提供了多套API.在今天的文章中将会对这些内容进行一一介绍: 音频 音

iOS开发-KVO的奥秘

iOS开发-KVO的奥秘 字数2933 阅读3842 评论28 喜欢57 序言 在iOS开发中,苹果提供了许多机制给我们进行回调.KVO(key-value-observing)是一种十分有趣的回调机制,在某个对象注册监听者后,在被监听的对象发生改变时,对象会发送一个通知给监听者,以便监听者执行回调操作.最常见的KVO运用是监听scrollView的contentOffset属性,来完成用户滚动时动态改变某些控件的属性实现效果,包括渐变导航栏.下拉刷新控件等效果. 渐变导航栏 使用 KVO的使用

iOS开发系列--让你的应用“动”起来

--iOS核心动画 概览 在iOS中随处都可以看到绚丽的动画效果,实现这些动画的过程并不复杂,今天将带大家一窥iOS动画全貌.在这里你可以看到iOS中如何使用图层精简非交互式绘图,如何通过核心动画创建基础动画.关键帧动画.动画组.转场动画,如何通过UIView的装饰方法对这些动画操作进行简化等.在今天的文章里您可以看到动画操作在iOS中是如何简单和高效,很多原来想做但是苦于没有思路的动画在iOS中将变得越发简单: CALayer CALayer简介 CALayer常用属性 CALayer绘图 C