在軟件開發(fā)過程中,我們常常需要在某個(gè)時(shí)間后執(zhí)行某個(gè)方法,或者是按照某個(gè)周期一直執(zhí)行某個(gè)方法。在這個(gè)時(shí)候,我們就需要用到定時(shí)器。
然而,在iOS中有很多方法完成以上的任務(wù),到底有多少種方法呢?經(jīng)過查閱資料,大概有三種方法:NSTimer、CADisplayLink、GCD。接下來我就一一介紹它們的用法。
一、NSTimer
1. 創(chuàng)建方法
NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self
selector:@selector(action:) userInfo:nil repeats:NO];
**TimerInterval **: 執(zhí)行之前等待的時(shí)間。比如設(shè)置成1.0,就代表1秒后執(zhí)行方法
target : 需要執(zhí)行方法的對象。
selector : 需要執(zhí)行的方法
repeats : 是否需要循環(huán)
-
2. 釋放方法
[timer invalidate]; 注意 :
調(diào)用創(chuàng)建方法后,target對象的計(jì)數(shù)器會加1,直到執(zhí)行完畢,自動減1。如果是循環(huán)執(zhí)行的話,就必須手動關(guān)閉,否則可以不執(zhí)行釋放方法。
3. 特性
- 存在延遲
不管是一次性的還是周期性的timer的實(shí)際觸發(fā)事件的時(shí)間,都會與所加入的RunLoop和RunLoop Mode有關(guān),如果此RunLoop正在執(zhí)行一個(gè)連續(xù)性的運(yùn)算,timer就會被延時(shí)出發(fā)。重復(fù)性的timer遇到這種情況,如果延遲超過了一個(gè)周期,則會在延時(shí)結(jié)束后立刻執(zhí)行,并按照之前指定的周期繼續(xù)執(zhí)行。
- 必須加入Runloop
使用上面的創(chuàng)建方式,會自動把timer加入MainRunloop的NSDefaultRunLoopMode中。如果使用以下方式創(chuàng)建定時(shí)器,就必須手動加入Runloop:
NSTimer *timer = [NSTimer timerWithTimeInterval:5 target:self selector:@selector(timerAction)
userInfo:nil repeats:YES];
[[NSRunLoop mainRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
- 注意
UIScrollView 拖動時(shí)執(zhí)行的是 UITrackingRunLoopMode,會導(dǎo)致暫停定時(shí)器,等恢復(fù)為 NSDefaultRunLoopMode 時(shí)才恢復(fù)定時(shí)器。 所以如果需要定時(shí)器在 UIScrollView 拖動時(shí)也不影響的話,建議添加到UITrackingRunLoopMode 或 NSRunLoopCommonModes 中:
NSTimer *timer = [NSTimer timerWithTimeInterval:5 target:self selector:@selector(timerAction)
userInfo:nil repeats:YES];
[[NSRunLoop mainRunLoop] addTimer:timer forMode: UITrackingRunLoopMode]; ///< 或者 NSRunLoopCommonModes
二、CADisplayLink
1. 創(chuàng)建方法
self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(handleDisplayLink:)];
[self.displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
2. 停止方法
[self.displayLink invalidate];
self.displayLink = nil;
/**當(dāng)把CADisplayLink對象add到runloop中后,selector就能被周期性調(diào)用,類似于重復(fù)的NSTimer被啟動了;
執(zhí)行invalidate操作時(shí),CADisplayLink對象就會從runloop中移除,selector調(diào)用也隨即停止,
類似于NSTimer的invalidate方法。**/
3. 特性
屏幕刷新時(shí)調(diào)用
CADisplayLink是一個(gè)能讓我們以和屏幕刷新率同步的頻率將特定的內(nèi)容畫到屏幕上的定時(shí)器類。CADisplayLink以特定模式注冊到runloop后,每當(dāng)屏幕顯示內(nèi)容刷新結(jié)束的時(shí)候,runloop就會向CADisplayLink指定的target發(fā)送一次指定的selector消息, CADisplayLink類對應(yīng)的selector就會被調(diào)用一次。所以通常情況下,按照iOS設(shè)備屏幕的刷新率60次/秒延遲
iOS設(shè)備的屏幕刷新頻率是固定的,CADisplayLink在正常情況下會在每次刷新結(jié)束都被調(diào)用,精確度相當(dāng)高。但如果調(diào)用的方法比較耗時(shí),超過了屏幕刷新周期,就會導(dǎo)致跳過若干次回調(diào)調(diào)用機(jī)會。
如果CPU過于繁忙,無法保證屏幕60次/秒的刷新率,就會導(dǎo)致跳過若干次調(diào)用回調(diào)方法的機(jī)會,跳過次數(shù)取決CPU的忙碌程度。
-
使用場景
從原理上可以看出,CADisplayLink適合做界面的不停重繪,比如視頻播放的時(shí)候需要不停地獲取下一幀用于界面渲染。
4. 重要屬性
frameInterval
NSInteger類型的值,用來設(shè)置間隔多少幀調(diào)用一次selector方法,默認(rèn)值是1,即每幀都調(diào)用一次。duration
readOnly的CFTimeInterval值,表示兩次屏幕刷新之間的時(shí)間間隔。需要注意的是,該屬性在target的selector被首次調(diào)用以后才會被賦值。selector的調(diào)用間隔時(shí)間計(jì)算方式是:調(diào)用間隔時(shí)間 = duration × frameInterval。
三、GCD方式
執(zhí)行一次
double delayInSeconds = 2.0;
dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, delayInSeconds * NSEC_PER_SEC);
dispatch_after(popTime, dispatch_get_main_queue(), ^(void){ 4 //執(zhí)行事件5 });
重復(fù)執(zhí)行
NSTimeInterval period = 1.0; //設(shè)置時(shí)間間隔
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); //每秒執(zhí)行
dispatch_source_set_event_handler(timer, ^{ //在這里執(zhí)行事件 });
dispatch_resume(timer);