iOS 定時器 NSTimer、CADisplayLink、GCD3種方式的實現,quartz定時器實現方式
在軟體開發過程中,我們常常需要在某個時間後執行某個方法,或者是按照某個周期一直執行某個方法。在這個時候,我們就需要用到定時器。
然而,在iOS中有很多方法完成以上的任務,到底有多少種方法呢?經過查閱資料,大概有三種方法:NSTimer、CADisplayLink、GCD。接下來我就一一介紹它們的用法。
一、NSTimer
1. 建立方法
1 NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(action:) userInfo:nil repeats:NO];
2. 釋放方法
1 [timer invalidate];
調用建立方法後,target對象的計數器會加1,直到執行完畢,自動減1。如果是迴圈執行的話,就必須手動關閉,否則可以不執行釋放方法。
3. 特性
不管是一次性的還是周期性的timer的實際觸發事件的時間,都會與所加入的RunLoop和RunLoop Mode有關,如果此RunLoop正在執行一個連續性的運算,timer就會被延時出發。重複性的timer遇到這種情況,如果延遲超過了一個周期,則會在延時結束後立刻執行,並按照之前指定的周期繼續執行。
使用上面的建立方式,會自動把timer加入MainRunloop的NSDefaultRunLoopMode中。如果使用以下方式建立定時器,就必須手動加入Runloop:
1 NSTimer *timer = [NSTimer timerWithTimeInterval:5 target:self selector:@selector(timerAction) userInfo:nil repeats:YES];2 [[NSRunLoop mainRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
UIScrollView 拖動時執行的是 UITrackingRunLoopMode,會導致暫停定時器,等恢複為 NSDefaultRunLoopMode 時才恢複定時器。
所以如果需要定時器在 UIScrollView 拖動時也不影響的話,建議添加到 UITrackingRunLoopMode 或 NSRunLoopCommonModes 中:
1 NSTimer *timer = [NSTimer timerWithTimeInterval:5 target:self selector:@selector(timerAction) userInfo:nil repeats:YES];2 [[NSRunLoop mainRunLoop] addTimer:timer forMode: UITrackingRunLoopMode]; ///< 或者 NSRunLoopCommonModes
二、CADisplayLink
1. 建立方法
1 self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(handleDisplayLink:)]; 2 [self.displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
2. 停止方法
1 [self.displayLink invalidate]; 2 self.displayLink = nil;3 4 /**當把CADisplayLink對象add到runloop中後,selector就能被周期性調用,類似於重複的NSTimer被啟動了;執行invalidate操作時,CADisplayLink對象就會從runloop中移除,selector調用也隨即停止,類似於NSTimer的invalidate方法。**/
3. 特性
螢幕重新整理時調用
CADisplayLink是一個能讓我們以和螢幕重新整理率同步的頻率將特定的內容畫到螢幕上的定時器類。CADisplayLink以特定模式註冊到runloop後,每當螢幕顯示內容重新整理結束的時候,runloop就會向CADisplayLink指定的target發送一次指定的selector訊息, CADisplayLink類對應的selector就會被調用一次。所以通常情況下,按照iOS裝置螢幕的重新整理率60次/秒
延遲
使用情境
從原理上可以看出,CADisplayLink適合做介面的不停重繪,比如視頻播放的時候需要不停地擷取下一幀用於介面渲染。
4. 重要屬性
三、GCD方式
執行一次
1 double delayInSeconds = 2.0;2 dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, delayInSeconds * NSEC_PER_SEC); 3 dispatch_after(popTime, dispatch_get_main_queue(), ^(void){ 4 //執行事件5 });
重複執行
1 NSTimeInterval period = 1.0; //設定時間間隔2 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);3 dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);4 dispatch_source_set_timer(_timer, dispatch_walltime(NULL, 0), period * NSEC_PER_SEC, 0); //每秒執行5 dispatch_source_set_event_handler(_timer, ^{6 //在這裡執行事件7 });8 dispatch_resume(_timer);
1.建立方法
//需要將dispatch_source_t timer設定為成員變數,不然會立即釋放@property (nonatomic, strong) dispatch_source_t timer;//定時器開始執行的延時時間NSTimeInterval delayTime = 3.0f;//定時器間隔時間NSTimeInterval timeInterval = 3.0f; //建立子線程隊列dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);//使用之前建立的隊列來建立計時器_timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);//設定延時執行時間,delayTime為要延時的秒數dispatch_time_t startDelayTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayTime * NSEC_PER_SEC));//設定計時器dispatch_source_set_timer(_timer, startDelayTime, timeInterval * NSEC_PER_SEC, 0.1 * NSEC_PER_SEC);dispatch_source_set_event_handler(_timer, ^{ //執行事件 });// 啟動計時器dispatch_resume(_timer)
2.停止方法
dispatch_source_cancel(_timer);
3.重要屬性
收藏學習,