iOS 定時器進入后臺繼續(xù)運行問題探索

一. 問題背景

最近項目中有個定時器計時實時更新等車的時長续挟,因為項目里面進入后臺是有執(zhí)行一些任務(wù)的操作跑芳,因此如果進入后臺時間不長博个,是定時器是不會暫停的往堡,但如果進入后臺時間投蝉,超過20s以上关拒,定時器就暫停谐算,回到前臺重新開始倒計時洲脂,這時候等車的時長會出現(xiàn)不準的情況。

二. 問題原因

經(jīng)驗證NSTimer,CADisplayLink一铅,dispatch_source_t,三個定時器,在進入到后臺的時候卜录,都會暫停艰毒,等到返回前臺的時候凯傲,才會繼續(xù)回調(diào)冰单。

timer.gif

看了一些博客說加上后臺任務(wù)執(zhí)行這句話可以保證App進入后臺浴栽,定時器不會暫停被廓,依然繼續(xù)執(zhí)行

[[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:nil];

經(jīng)驗證嫁乘,后臺執(zhí)行任務(wù)也將暫停延遲蜓斧,還是沒辦法解決App長時間進入后臺,定時器暫停問題直奋。

我們通過監(jiān)聽mainRunLoop回調(diào)可以發(fā)現(xiàn)帮碰,當App進入到后臺殉挽,mainRunLoop進入了休眠,當App回到前臺傻唾,mainRunLoop重新喚醒繼續(xù)執(zhí)行伪煤。

main_runloop_timer.gif

因此我再想,如果在App進入后臺的時候防泵,將已經(jīng)睡眠的mainRunLoop重新喚醒捷泞,是不是就可以保證定時器的不暫停,持續(xù)運行骡湖。

- (void)didEnterBackground {
    NSLog(@"--------------------------didEnterBackground");
    [[NSRunLoop mainRunLoop] run];
}

main_runloop_background_run_timer.gif

經(jīng)驗證惠桃,結(jié)果如猜想一樣辜王,在App進入后臺,重新喚醒mainRunLoop汹来,可以保證定時器不暫停收班,可以一直運行。
但詭異的事情發(fā)生了邻耕,當App重新回到前臺兄世,willEnterForeground回調(diào)也沒走碘饼,點擊返回按鍵竟然沒有響應(yīng)艾恼。這究竟為什么呢舆声?
找了一些資料才發(fā)現(xiàn):
[[NSRunLoop mainRunLoop] run];并不是喚醒mainRunLoop,而是會使得主線程陷入休眠蛾找,永遠等待打毛,但會讓出主線程時間片。

[[NSRunLoop mainRunLoop] run]; //主線程永遠等待熬甫,但讓出主線程時間片
[[NSRunLoop mainRunLoop] runUntilDate:[NSDate distantFuture]]; //等同上面調(diào)用
[[NSRunLoop mainRunLoop] runUntilDate:[NSDate date]]; //立即返回
[[NSRunLoop mainRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:10.0]]; //主線程等待,但讓出主線程時間片覆旱,然后過10秒后返回
[[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate: [NSDate distantFuture]]; //主線程等待,但讓出主線程時間片炼彪;有事件到達就返回辐马,比如點擊UI等。
[[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate: [NSDate date]]; //立即返回
[[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate: [NSDate dateWithTimeIntervalSinceNow:10.0]]; //主線程等待檩帐,但讓出主線程時間片;有事件到達就返回四敞,如果沒有則過10秒返回达箍。

因此這里我推斷缎玫,進入后臺之后,調(diào)用[[NSRunLoop mainRunLoop] run];使得主線程陷入休眠,永遠等待,導(dǎo)致主線程沒有去處理定時器相關(guān)停止操作缕减,因此定時器才能繼續(xù)執(zhí)行。

那有沒有辦法可以等回到前臺的時候部逮,喚醒主線程的呢兄朋?

答案是沒有,因為主線程休眠怜械,導(dǎo)致所有前后臺相關(guān)的通知都不再回調(diào)颅和,因此也就沒法區(qū)分App是否回到前臺,也沒辦法去喚醒主線程缕允。

那有沒有方法保證App進入后臺之后峡扩,定時器依然能繼續(xù)執(zhí)行呢?

這個問題我也跟朋友探討了下灼芭,正常情況下寄悯,App進入后臺之后洒擦,如果沒有申請持續(xù)更新定位椅邓、或者語音等權(quán)限绰精,也沒有通過UIBackgroundTaskIdentifier向系統(tǒng)借用一段時間,這時候正常App進入后臺就會去中斷其他任務(wù)的執(zhí)行门岔,包括定時任務(wù),從而掛起好渠,保證系統(tǒng)流暢運行拳锚。

這也意味著棋凳,要先保證App進入后臺,定時器依然能順利執(zhí)行尚困,就要先保證App進入后臺依然存活,而不會掛起邻悬。

而保證App進入后臺的方法,無非就是勾選并實現(xiàn)對應(yīng)的后臺模式蘑斧。

除此之外也有一些取巧的方法花颗,如:

  1. App退到后臺
  2. 使用beginBackgroundTaskWithExpirationHandler向系統(tǒng)申請3分鐘的后臺任務(wù),此時backgroundTimeRemaining=180
  3. 3分鐘限制快到時域携,啟動定位旋圆,此時3分鐘限制被打破,此時backgroundTimeRemaining=DBL_MAX
  4. 一小段時間后(如2s),停止定位屋匕,此時backgroundTimeRemaining≈0
  5. 重復(fù)步驟2

這些方法都是一些取巧的方式翩隧,若非必要,不建議采取。

雖然App進入后臺后茂翔,定時器會暫停,但是當App回到前臺時迁沫,定時器會立馬回調(diào)。因此針對計時類的需求祈餐,可以在App定時器啟動之前绷杜,記錄一個當前系統(tǒng)時間的時間戳preTime,當定時器每次回調(diào)歇竟,就取當前系統(tǒng)時間戳curTime懊烤,然后計算兩個時間戳的差值difTime弃秆,然后用需要倒計時的時間totalTime们拙,減去時間戳差值difTime,就可以算出倒計時后的時間,這樣就能優(yōu)雅的解決坷虑,倒計時進入后臺不準確的問題账磺。

三. 結(jié)論

關(guān)于iOS定時器進入后臺后在不開啟相關(guān)后臺模式氏捞,比如持續(xù)定位更新辞嗡、音樂播放等模式下,要想保證定時器進入后臺能持續(xù)執(zhí)行,我目前并沒有找到除了一些取巧方法后的好的通用方法达址。
因此這里對于一些定時器倒計時進入后臺不準確問題苛败,推薦使用如下方案:

App定時器啟動之前,記錄一個當前系統(tǒng)時間的時間戳preTime译蒂,當定時器每次回調(diào)炎辨,就取當前系統(tǒng)時間戳curTime,然后計算兩個時間戳的差值difTime,然后用需要倒計時的時間totalTime曹货,減去時間戳差值difTime,就可以算出倒計時后的時間,這樣就能優(yōu)雅的解決,倒計時進入后臺不準確的問題。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末剃诅,一起剝皮案震驚了整個濱河市如筛,隨后出現(xiàn)的幾起案子杨刨,更是在濱河造成了極大的恐慌晤柄,老刑警劉巖芥颈,帶你破解...
    沈念sama閱讀 216,324評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件署辉,死亡現(xiàn)場離奇詭異,居然都是意外死亡耕姊,警方通過查閱死者的電腦和手機桶唐,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,356評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來邦邦,“玉大人安吁,你說我怎么就攤上這事∪枷剑” “怎么了鬼店?”我有些...
    開封第一講書人閱讀 162,328評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長黔龟。 經(jīng)常有香客問我妇智,道長滥玷,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,147評論 1 292
  • 正文 為了忘掉前任巍棱,我火速辦了婚禮惑畴,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘航徙。我一直安慰自己如贷,他們只是感情好,可當我...
    茶點故事閱讀 67,160評論 6 388
  • 文/花漫 我一把揭開白布到踏。 她就那樣靜靜地躺著杠袱,像睡著了一般。 火紅的嫁衣襯著肌膚如雪窝稿。 梳的紋絲不亂的頭發(fā)上楣富,一...
    開封第一講書人閱讀 51,115評論 1 296
  • 那天,我揣著相機與錄音伴榔,去河邊找鬼纹蝴。 笑死,一個胖子當著我的面吹牛踪少,可吹牛的內(nèi)容都是我干的塘安。 我是一名探鬼主播,決...
    沈念sama閱讀 40,025評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼秉馏,長吁一口氣:“原來是場噩夢啊……” “哼耙旦!你這毒婦竟也來了脱羡?” 一聲冷哼從身側(cè)響起萝究,我...
    開封第一講書人閱讀 38,867評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎锉罐,沒想到半個月后帆竹,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,307評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡脓规,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,528評論 2 332
  • 正文 我和宋清朗相戀三年栽连,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片侨舆。...
    茶點故事閱讀 39,688評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡秒紧,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出挨下,到底是詐尸還是另有隱情熔恢,我是刑警寧澤,帶...
    沈念sama閱讀 35,409評論 5 343
  • 正文 年R本政府宣布臭笆,位于F島的核電站叙淌,受9級特大地震影響秤掌,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜鹰霍,卻給世界環(huán)境...
    茶點故事閱讀 41,001評論 3 325
  • 文/蒙蒙 一闻鉴、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧茂洒,春花似錦孟岛、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,657評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至玷氏,卻和暖如春堵未,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背盏触。 一陣腳步聲響...
    開封第一講書人閱讀 32,811評論 1 268
  • 我被黑心中介騙來泰國打工渗蟹, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人赞辩。 一個月前我還...
    沈念sama閱讀 47,685評論 2 368
  • 正文 我出身青樓雌芽,卻偏偏與公主長得像俄认,于是被迫代替她去往敵國和親曙博。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,573評論 2 353

推薦閱讀更多精彩內(nèi)容