iOS GCD封裝Timer

歡迎加QQ群: 457236811 ,我們一起來搞基!

一、為啥要用GCD封裝Timer

  • 為什么要用GCD封裝Timer呢?主要是系統(tǒng)提供的NSTimer可能不準,有同學就問為啥蘋果提供的Timer為啥就不準了吖?這么麻煩要自己造

  • 其實啊 主要還是NSTimer是基于NSRunloop封裝的,這個鬼東西是啥?感興趣的同學可以去查查,我這里就把大概說一下

1.NSRunloop有兩種Mode 分別為NSDefaultRunLoopMode、NSRunLoopCommonModes當她切換currentMode的時候,Timer會失效(就是不會跑了)
2.NSRunloop她相當于一個循環(huán),她很忙的時候要一圈,一圈的跑,這樣她就跑到不均勻,繞圈圈的時間不一樣,而且跑每圈的固定位置才處理Timer,所以NSTimer可能因為NSRunloop太忙就不準(主要原因)
3.那么GCD封裝的好處呢? GCD基于系統(tǒng)內(nèi)核管理,不依賴于NSRunloop,而且聽說能精確到納秒級別

二、用GCD封裝Timer

  • 說了那么多終于可以開干了(怎么這么啰嗦),直接上代碼吧:
/// 初始化timer 返回timer唯一標識符
/// @param task 回調(diào)block
/// @param start 多少秒后開始
/// @param interval 定時器間隔時間
/// @param repeats 是否重復
/// @param async 是否在異步線程創(chuàng)建timer
+(NSString *)execTask:(void(^)(void))task
                start:(NSTimeInterval)start
             interval:(NSTimeInterval)interval
              repeats:(BOOL)repeats
                async:(BOOL)async;


/// 取消對應(yīng)的timr
/// @param name timer唯一標識符
+ (void)cancelTask:(NSString *)name;
  • .m文件代碼實現(xiàn):
//存放定時器的全局字典
static NSMutableDictionary *timers;
// 信號量加鎖
dispatch_semaphore_t semaphore;

+ (void)initialize {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        semaphore = dispatch_semaphore_create(1);
        timers = [NSMutableDictionary dictionary];
    });
}

+ (NSString *)execTask:(void (^)(void))task
                 start:(NSTimeInterval)start
              interval:(NSTimeInterval)interval
               repeats:(BOOL)repeats
                 async:(BOOL)async
{
    
    if (!task || interval<=0 || start<0) return nil;
    
    static int i = 0;
    
    // 定時器唯一標識符
    NSString *identify = [NSString stringWithFormat:@"%d",i++];
        
    // 隊列
    dispatch_queue_t queue = async ? dispatch_get_global_queue(0, 0) : dispatch_get_main_queue();
    
    // 創(chuàng)建定時器
    dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);
    
    // 設(shè)置時間
    dispatch_source_set_timer(timer, dispatch_time(DISPATCH_TIME_NOW, start * NSEC_PER_SEC), interval * NSEC_PER_SEC, 0);
    
    // 設(shè)置回調(diào)
    dispatch_source_set_event_handler(timer,^{
        
        task();
        if (!repeats) {
            [self cancelTask:identify];
        }
    });
    
    // 啟動定時器
    dispatch_resume(timer);
    
    // 加鎖
    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    // 存放到全局字典
    timers[identify] = timer;
    // 解鎖
    dispatch_semaphore_signal(semaphore);

    return identify;
}

+ (void)cancelTask:(NSString *)name
{
    if (name.length > 0) {
        // 加鎖
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
        dispatch_source_cancel(timers[name]);
        // 移除對應(yīng)的timer
        [timers removeObjectForKey:name];
        // 解鎖
        dispatch_semaphore_signal(semaphore);
    }

}
  • 最后里面有注釋,但是有些東西還是要講一講的,這樣才好嘛

1.為什么用全局字典存放timer?方便取出來取消Timer(AFN SDWebImage 也是這么干)
2.為什么用dispatch_semaphore加鎖,NSMutableDictionary多線程下不安全 而且dispatch_semaphore加鎖速度快 (那個YYKit 大神有個測評,自旋鎖不安全,互斥鎖這個最快的)嗯 就是這樣

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 1.NSTimer不準時的原因:(1).RunLoop循環(huán)處理時間,每次循環(huán)是固定時間,只有在這段時間才會去查看N...
    稻春閱讀 1,360評論 0 3
  • 這周就學了多線程,現(xiàn)在還在云里霧里.... 線程的概念 執(zhí)行調(diào)度單位 并發(fā)資源訪問控制原子操作 Atomic(任務(wù)...
    逸飛u閱讀 520評論 0 1
  • 概述 RunLoop作為iOS中一個基礎(chǔ)組件和線程有著千絲萬縷的關(guān)系,同時也是很多常見技術(shù)的幕后功臣。盡管在平時多...
    sumrain_cloud閱讀 1,006評論 0 5
  • 原文鏈接:https://blog.cnbluebox.com/blog/2014/07/01/cocoashen...
    MxlZlh閱讀 248評論 0 0
  • 1. 并行和并發(fā) 簡單來說,若說兩個任務(wù)A和B并發(fā)執(zhí)行,則表示任務(wù)A和任務(wù)B在同一時間段里被執(zhí)行(更多的可能是二者...
    Z_Han閱讀 712評論 0 8

友情鏈接更多精彩內(nèi)容