頁面實現多個定時器(計時器)時選用NSTimer還是GCD?(乾貨不濕)

来源:http://www.cnblogs.com/beckwang0912/archive/2017/06/16/7027484.html
-Advertisement-
Play Games

定時器在我們每個人做的iOS項目裡面必不可少,如登錄頁面倒計時、支付期限倒計時等等,一般來說使用NSTimer創建定時器: + (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)a ...


定時器在我們每個人做的iOS項目裡面必不可少,如登錄頁面倒計時、支付期限倒計時等等,一般來說使用NSTimer創建定時器:

+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;

But 使用NSTimer需要註意一下幾點:

     1、必須保證有一個活躍的RunLoop。

     系統框架提供了幾種創建NSTimer的方法,其中以scheduled開頭的方法會自動把timer加入當前RunLoop,到了設定時間就會觸發selector方法,而沒有scheduled開頭的方法則需要手動添加timer到一個RunLoop中才會有效。程式啟動時,會預設啟動主線程的RunLoop併在程式運行期內有效,所以把timer放入主線程時不需要啟動RunLoop,但現實開發中主線程更多的是處理UI事物,把耗時且耗能的操作放在子線程中,這就需要將子線程的RunLoop激活。

     我們不難知道RunLoop在運行時一般有兩個:NSDefaultRunLoopMode、NSEventTrackingRunLoopMode,scheduled生成的timer會預設添加到NSDefaultRunLoopMode,當某些UI事件發生時,如頁面滑動RunLoop切換到NSEventTrackingRunLoopMode運行,我們會發現定時器失效,為瞭解決timer失效的問題,我們需要在scheduled一個定時器的時候,設置它的運行模式為:

     [[NSRunLoop currentRunLoop] addTimer:self.progressTimer forMode:NSRunLoopCommonModes];

     註意:NSRunLoopCommonModes並不是一種正在存在的運行狀態,這個模式等效於NSDefaultRunLoopMode和NSEventTrackingRunLoopMode的結合,相當於它標記了timer可以在這兩種模式下都有效。

     2.NSTimer的創建與撤銷必須在同一個線程操作,不能跨越線程操作。

     3.存在記憶體泄漏的風險(這個問題需要引起重視)

     scheduledTimerWithTimeInterval方法將target設為A對象時,A對象會被這個timer所持有,也就是會被retain一次,timer又會被當前的runloop所持有。使用NSTimer時,timer會保持對target和userInfo參數的強引用。只有當調取了NSTimer的invalidate方法時,NSTimer才會釋放target和userInfo。生成timer的方法中如果repeats參數為NO,則定時器觸發後會自動調取invalidate方法。如果repeats參數為YES,則需要手動調取invalidate方法才能釋放timer對target和userIfo的強引用。

    - (void)cancel{

          [_timer invalidate];

           _timer = nil;

     }

    這裡要特別註意的一點是,按照各種資料顯示,我們在銷毀或者釋放對象時,大部分都是在dealloc方法中,然後我們高高興興的在dealloc里寫上

   - (void)dealloc{

         [self cancel];

    }

   以為這樣就可以釋放timer了,不幸的是,dealloc方法永遠不會被調用。因為timer的引用,對象A的引用計數永遠不會降到0,這時如果不調用cancel,對象X將永遠無法釋放,造成記憶體泄露。所以我建議在使用定時器的事件完成後立即將timer進行cancel,如果是比較長時間的定時器,可以在頁面消失事件中調用,如:

   - (void)viewWillDisappear:(BOOL)animated{

    [super viewWillDisappear:animated];

    [self cancel];

   }

   看到這裡,你會不會發現使用NSTimer實現定時器這麼麻煩,又是RunLoop,又是線程的,一會兒還得考慮記憶體泄露,So , 如果在一個頁面需要同時顯示多個計時器的時候,NSTimer簡直就是災難了。那麼有沒有高逼格的辦法實現呢?答案就是GCD!  以下5點是使用dispatch_source_t創建timer的主要知識點:

   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.設置首次執行事件、執行間隔和精確度

      dispatch_source_set_timer(timer, dispatch_time(DISPATCH_TIME_NOW, interval * NSEC_PER_SEC), interval * NSEC_PER_SEC, 0.1 * NSEC_PER_SEC);

   4.處理事件block

    dispatch_source_set_event_handler(timer, ^{

            // doSomething()

    });

    5.激活timer / 取消timer

    dispatch_resume(timer);   /    dispatch_source_cancel(timer);

    寫到這裡,自然要問如果我只是想執行一次,不需要迴圈實現定時器那怎麼辦呢?那也沒問題,參考NSTimer,我們可以集成repeats選項,當repeats = No時,在激活timer並回調block事件後dispatch_source_cancel掉當前dispatch_source_t  timer即可,如下所示:

      

   

    上面的代碼就創建了一個timer,如果repeats = NO,在一個周期完成後,系統會自動cancel掉這個timer;如果repeats=YES,那麼timer會一個周期接一個周期的執行,直到你手動cancel掉這個timer,你可以在dealloc方法裡面做cancel,這樣timer恰好運行於整個對象的生命周期中。這裡不必要擔心NSTimer因dealloc始終無法調而產生的記憶體泄漏問題,你也可以通過queue參數控制這個timer所添加到的線程,也就是action最終執行的線程。傳入nil則會預設放到子線程中執行。UI相關的操作需要傳入dispatch_get_main_queue()以放到主線程中執行。

    寫到這裡,基本上可以滿足開發要求,然而我們可以更加變態,假設這樣的場景,每次開始新一次的計時前,需要取消掉上一次的計時任務 或者 將上一次計時的任務,合併到新的一次計時中,最終一併執行!針對這兩種場景,也已經集成到上面的介面scheduleGCDTimerWithName中。具體代碼請看demo!

github地址:https://github.com/BeckWang0912/ZTGCDTimer

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這篇是講Glide源碼中into方法的實現原理,可以說with和load方法只是做了前期的初始化配置工作,而真正意義上的圖片載入就是在into方法中實現的,所以該方法的複雜程度是可以想象的,還是依照我之前的寫作習慣,一步步的分析,不留下任何的盲點給大家帶來困惑,那麼下麵就開始吧。 ...
  • 我們都知道系統要確定View的大小,首先得先獲得MeasureSpec,再通過MeasureSpec來決定View的大小。 MeasureSpec(32為int值)由兩部分組成: SpecMode(高2位):測量模式。 SpecSize(低30位):某種測量模式下的規格大小。 SpecMode有3類 ...
  • 在你的程式沒有發佈(release)到AppStore的時候,你可以使用TestFlight去邀請用戶測試你的app,進而收集有用的反饋信息。 ...
  • 創建動畫 協議代理 屬性 設置動畫組件Item的動力屬性 UIDynamicItemBehavior 為動畫組件添加具體行為 吸引行為 UISnapBehavior 重力行為 UIGravityBehavior 碰撞行為 UICollisionBehavior 作用力行為 UIPushBehavio ...
  • 前幾天要用到PopupWindow,一時竟想不起來怎麼用,趕緊上網查了查,自己寫了個demo,併在此記錄一下PopupWindow的用法。 使用場景 PopupWindow,顧名思義,就是彈窗,在很多場景下都可以見到它。例如ActionBar/Toolbar的選項彈窗,一組選項的容器,或者列表等集合 ...
  • 蘋果手機有個痛點,就是不支持文件管理。對於那些需要在移動端處理文檔或者說整理文件的朋友來說,沒有文件管理這個功能絕對讓人抓狂,好在作為良心企業的Readdle,為我們開發了Documents,並且免費。目前的版本是Documents 6,可以在App Store免費下載。 這個文件管理器支持的文檔格 ...
  • 要理解View的測量,首先要瞭解MeasureSpec,系統在測量view的寬高時,要先確定MeasureSpec。 MeasureSpec(32為int值)由兩部分組成: SpecMode(高2位):測量模式。 SpecSize(低30位):某種測量模式下的規格大小。 SpecMode有3類: U ...
  • View事件的分發機制由三個方法共同完成,這三個方法是: public boolean dispatchTouchEvent(MotionEvent ev); public boolean onInterceptTouchEvent(MotionEvent ev); public boolean o ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...