前言
在開發中,很多時候我們需要用到定時器實時刷新某個數值。這個時候我們就需要用到定時器,這裡,我為大家推薦三種方法,分別是:NSTimer、CADisplayLink、GCD。接下來我就一一介紹它們的用法。希望能幫到大家。
一、NSTimer(一般用於定時的更新一些非界面上的數據)
1. 創建方法
NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(action:) userInfo:nil repeats:NO];
TimerInterval
: 執行之前等待的時間。比如設置成1.0,就代表1秒後執行方法
target
: 需要執行方法的對象。
selector
: 需要執行的方法
repeats
: 是否需要循環
2. 釋放方法
[timer invalidate];
注意 :
調用創建方法後,target
對象的計數器會加1,直到執行完畢,自動減1。如果是循環執行的話,就必須手動關閉,否則可以不執行釋放方法。
3. 特性
存在延遲
不管是一次性的還是周期性的timer
的實際觸發事件的時間,都會與所加入的RunLoop
和RunLoop Mode
有關,如果此RunLoop
正在執行一個連續性的運算,timer
就會被延時出發。重復性的timer
遇到這種情況,如果延遲超過了一個周期,則會在延時結束後立刻執行,並按照之前指定的周期繼續執行。
必須加入Runloop
使用上面的創建方式,會自動把timer
加入MainRunloop
的NSDefaultRunLoopMode
中。如果使用以下方式創建定時器,就必須手動加入Runloop
:
NSTimer *timer = [NSTimer timerWithTimeInterval:5 target:self selector:@selector(timerAction) userInfo:nil repeats:YES]; [[NSRunLoop mainRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
二、CADisplayLink
1. 創建方法
self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(handleDisplayLink:)]; [self.displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
2. 停止方法
self.displayLink invalidate]; self.displayLink = nil;
當把CADisplayLink對象add到runloop
中後,selector
就能被周期性調用,類似於重復的NSTimer被啟動了;執行invalidate
操作時,CADisplayLink對象就會從runloop
中移除,selector
調用也隨即停止,類似於NSTimer的invalidate
方法。
3. 特性
屏幕刷新時調用
CADisplayLink是一個能讓我們以和屏幕刷新率同步的頻率將特定的內容畫到屏幕上的定時器類。CADisplayLink以特定模式注冊到runloop
後,每當屏幕顯示內容刷新結束的時候,runloop
就會向CADisplayLink指定的target
發送一次指定的selector
消息, CADisplayLink類對應的selector
就會被調用一次。所以通常情況下,按照iOS設備屏幕的刷新率60次/秒
延遲
iOS設備的屏幕刷新頻率是固定的,CADisplayLink在正常情況下會在每次刷新結束都被調用,精確度相當高。但如果調用的方法比較耗時,超過了屏幕刷新周期,就會導致跳過若干次回調調用機會。
如果CPU過於繁忙,無法保證屏幕60次/秒的刷新率,就會導致跳過若干次調用回調方法的機會,跳過次數取決CPU的忙碌程度。
使用場景
從原理上可以看出,CADisplayLink適合做界面的不停重繪,比如視頻播放的時候需要不停地獲取下一幀用於界面渲染。
4. 重要屬性
frameInterval
NSInteger類型的值,用來設置間隔多少幀調用一次selector
方法,默認值是1,即每幀都調用一次。
duration
readOnly
的CFTimeInterval
值,表示兩次屏幕刷新之間的時間間隔。需要注意的是,該屬性在target
的selector
被首次調用以後才會被賦值。selector
的調用間隔時間計算方式是:調用間隔時間 = duration × frameInterval
。
三、GCD方式
執行一次
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){ //執行事件 });
重復執行
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的方式,我在網上只能找到這些資料,目前自己還在學習中,以後會更新,以上就是iOS開發定時器的三種方式,希望本文能對各位iOS開發者們有所幫助,如果有疑問大家可以留言交流。