NSRunloop卡頓監(jiān)控

說(shuō)說(shuō)界面卡頓是怎么產(chǎn)生的食呻?
先說(shuō)屏幕流炕,蘋(píng)果移動(dòng)設(shè)備屏幕,即顯示器的刷新頻率是60HZ仅胞,這是硬件設(shè)備決定的每辟,無(wú)論使用者感覺(jué)卡還是不卡,都會(huì)按照這個(gè)頻率進(jìn)行刷新干旧。顯示器顯示的內(nèi)容是由顯卡渲染的渠欺,顯卡渲染一幀并顯示到顯示器上的時(shí)間點(diǎn),程序可以通過(guò)CADisplayLink捕獲椎眯。由于iOS設(shè)備都開(kāi)啟了垂直同步挠将,顯卡總是等到顯示器發(fā)出垂直同步信號(hào)后再開(kāi)始渲染下一幀。如果兩次垂直同步信號(hào)之間编整,即16.7ms內(nèi)舔稀,渲染數(shù)據(jù)沒(méi)有準(zhǔn)備好,那么這一幀數(shù)據(jù)就會(huì)丟失掌测,顯示器刷新的仍然是上一幀的數(shù)據(jù)内贮,造成掉幀卡頓。

那么什么情況下會(huì)導(dǎo)致沒(méi)有準(zhǔn)備好渲染數(shù)據(jù)呢?
這需要考慮渲染數(shù)據(jù)從哪來(lái)夜郁。App主線程在CPU中計(jì)算顯示內(nèi)容什燕,比如視圖的創(chuàng)建、布局計(jì)算拂酣、圖片解碼秋冰、文本繪制等。隨后CPU會(huì)將計(jì)算好的內(nèi)容提交到GPU去婶熬,由GPU進(jìn)行變換剑勾、合成、渲染赵颅。隨后GPU會(huì)把渲染結(jié)果提交到幀緩沖區(qū)去虽另。CPU和GPU不論哪個(gè)阻礙了顯示流程,都會(huì)造成掉幀現(xiàn)象饺谬。

我們?cè)诔绦蛑心茏龅闹挥斜O(jiān)控CPU了捂刺,GPU無(wú)能為力,而且通過(guò)觀察instruments募寨,會(huì)發(fā)現(xiàn)除了離屏渲染族展,其他情況下GPU并不是瓶頸,平時(shí)開(kāi)發(fā)中盡量避免即可拔鹰。主線程上的CPU工作都是在RunLoop中進(jìn)行的仪缸,從下面的偽代碼可以看到主要計(jì)算工作都在kCFRunLoopAfterWaiting和下一次kCFRunLoopBeforeWaiting之間。

setupThisRunLoopRunTimeoutTimer(); //by GCD timer
__CFRunLoopDoObservers(KCFRunLoopEntry);
do
{
__CFRunLoopDoObservers(kCFRunLoopBeforeTimers);
__CFRunLoopDoObservers(kCFRunLoopBeforeSources);

__CFRunLoopDoBlocks(); //處理非延遲的主線程調(diào)用
__CFRunLoopDoSource0(); //處理UIEvent事件

CheckIfExistMessagesInMainDispatchQueue(); //檢查GCD是否有在MainDispatchQueue中要執(zhí)行的事件
__CFRunLoopDoObservers(kCFRunLoopBeforeWaiting);

mach_port_t wakeUpPort = SleepAndWaitForWakingUpPorts(); //等待內(nèi)核mach_msg事件
//Zzz...
__CFRunLoopDoObservers(kCFRunLoopAfterWaiting);

//處理喚醒事件
if (wakeUpPort == timerPort) //timer喚醒
    __CFRunLoopDoTimers();
else if (wakeUpPort == mainDispatchQueuePort) //GCD喚醒
    __CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__()
else //端口喚醒列肢,如網(wǎng)絡(luò)請(qǐng)求回來(lái)
    __CFRunLoopDoSource1();

__CFRunLoopDoBlocks();
} while (!stop && !timeout);
__CFRunLoopDoObservers(CFRunLoopExit);

所以恰画,可以將監(jiān)控的RunLoop的運(yùn)行時(shí)間,設(shè)置為kCFRunLoopAfterWaiting開(kāi)始到下一次kCFRunLoopBeforeWaiting結(jié)束瓷马。這雖然和系統(tǒng)意義上一次完整的RunLoop不同(系統(tǒng)意義上的一次RunLoop拴还,應(yīng)該和AutoreleasePool的重建時(shí)機(jī)一樣,即kCFRunLoopBeforeWaiting到下一次kCFRunLoopBeforeWaiting之間)欧聘,但是runloop休眠的時(shí)間肯定不能認(rèn)為是卡頓片林。粗略計(jì)算一下,以運(yùn)行時(shí)低于40幀為卡怀骤,則會(huì)掉20幀费封,卡住的時(shí)間約為320ms∩古纾可以假設(shè)如果runloop執(zhí)行超過(guò)了0.3孝偎,主線程無(wú)法將計(jì)算好的內(nèi)容提交給 GPU访敌,會(huì)造成卡頓凉敲。

綜上,為主線程的 RunLoop 添加一個(gè) Observer ,來(lái)檢測(cè) RunLoop 的運(yùn)行情況爷抓。

CFRunLoopObserverContext context = {0, NULL, NULL, NULL};
CFRunLoopObserverRef observer = CFRunLoopObserverCreate(kCFAllocatorDefault,
                                                    kCFRunLoopAllActivities,
                                                    YES,
                                                    0,
                                                    &runLoopObserverCallBack,
                                                    &context);
CFRunLoopAddObserver(CFRunLoopGetMain(), observer, kCFRunLoopCommonModes);

在回調(diào)中势决,使用mach_absolute_time()記錄kCFRunLoopAfterWaiting的時(shí)間點(diǎn),在下一次kCFRunLoopBeforeWaiting時(shí)計(jì)算RunLoop的運(yùn)行時(shí)間蓝撇,如果超時(shí)果复,可以根據(jù)需求處理,比如dump函數(shù)堆棧渤昌,并上傳監(jiān)控服務(wù)器等虽抄。示例中使用的是斷言處理。

static const NSTimeInterval kRunLoopThreshold = 0.3;
static uint64_t kStartTime = 0;
static void runLoopObserverCallBack(CFRunLoopObserverRef observer, CFRunLoopActivity activity, void *info)
{
  switch (activity) {
    case kCFRunLoopAfterWaiting:
        kStartTime = mach_absolute_time();
        break;
    case kCFRunLoopBeforeWaiting:
        if (kStartTime != 0 ) {
            uint64_t elapsed = mach_absolute_time() - kStartTime;
            mach_timebase_info_data_t timebase;
            mach_timebase_info(&timebase);
            NSTimeInterval duration = elapsed * timebase.numer / timebase.denom / 1e9;
            if (duration > kRunLoopThreshold) { 
                assert(0);
            }
        }
        break;
    default:
        break;
    }
}

上述計(jì)算中独柑,在kCFRunLoopBeforeWaiting時(shí)每次都需要將mach_absolute_time()的時(shí)間轉(zhuǎn)換成秒迈窟,會(huì)比較浪費(fèi),可以通過(guò)context參數(shù)傳進(jìn)來(lái)忌栅。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末车酣,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子索绪,更是在濱河造成了極大的恐慌湖员,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,865評(píng)論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件瑞驱,死亡現(xiàn)場(chǎng)離奇詭異娘摔,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)钱烟,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,296評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)晰筛,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人拴袭,你說(shuō)我怎么就攤上這事读第。” “怎么了拥刻?”我有些...
    開(kāi)封第一講書(shū)人閱讀 169,631評(píng)論 0 364
  • 文/不壞的土叔 我叫張陵怜瞒,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我般哼,道長(zhǎng)吴汪,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 60,199評(píng)論 1 300
  • 正文 為了忘掉前任蒸眠,我火速辦了婚禮漾橙,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘楞卡。我一直安慰自己霜运,他們只是感情好脾歇,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,196評(píng)論 6 398
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著淘捡,像睡著了一般藕各。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上焦除,一...
    開(kāi)封第一講書(shū)人閱讀 52,793評(píng)論 1 314
  • 那天激况,我揣著相機(jī)與錄音,去河邊找鬼膘魄。 笑死乌逐,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的创葡。 我是一名探鬼主播黔帕,決...
    沈念sama閱讀 41,221評(píng)論 3 423
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼蹈丸!你這毒婦竟也來(lái)了成黄?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 40,174評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤逻杖,失蹤者是張志新(化名)和其女友劉穎奋岁,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體荸百,經(jīng)...
    沈念sama閱讀 46,699評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡闻伶,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,770評(píng)論 3 343
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了够话。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蓝翰。...
    茶點(diǎn)故事閱讀 40,918評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖女嘲,靈堂內(nèi)的尸體忽然破棺而出畜份,到底是詐尸還是另有隱情,我是刑警寧澤欣尼,帶...
    沈念sama閱讀 36,573評(píng)論 5 351
  • 正文 年R本政府宣布爆雹,位于F島的核電站,受9級(jí)特大地震影響愕鼓,放射性物質(zhì)發(fā)生泄漏钙态。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,255評(píng)論 3 336
  • 文/蒙蒙 一菇晃、第九天 我趴在偏房一處隱蔽的房頂上張望册倒。 院中可真熱鬧,春花似錦磺送、人聲如沸驻子。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,749評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)拴孤。三九已至,卻和暖如春甲捏,著一層夾襖步出監(jiān)牢的瞬間演熟,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,862評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工司顿, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留芒粹,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,364評(píng)論 3 379
  • 正文 我出身青樓大溜,卻偏偏與公主長(zhǎng)得像化漆,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子钦奋,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,926評(píng)論 2 361

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

  • 這篇文章會(huì)非常詳細(xì)的分析 iOS 界面構(gòu)建中的各種性能問(wèn)題以及對(duì)應(yīng)的解決思路座云,同時(shí)給出一個(gè)開(kāi)源的微博列表實(shí)現(xiàn),通過(guò)...
    RobinYu閱讀 1,495評(píng)論 0 2
  • 如何讓iOS 保持界面流暢璧帝?這些技巧你知道嗎如何讓iOS 保持界面流暢?這些技巧你知道嗎 作者:ibireme這篇...
    seonhiu閱讀 926評(píng)論 0 7
  • 作為一個(gè)iOS小菜鳥(niǎo)富寿,當(dāng)我們需求做完之后睬隶,我們?cè)摳墒裁矗慨?dāng)然是學(xué)習(xí)页徐!最近看了很多關(guān)于iOS性能優(yōu)化的文章苏潜,為了便于...
    小蝦啦閱讀 2,947評(píng)論 1 4
  • 這篇文章會(huì)非常詳細(xì)的分析 iOS 界面構(gòu)建中的各種性能問(wèn)題以及對(duì)應(yīng)的解決思路,同時(shí)給出一個(gè)開(kāi)源的微博列表實(shí)現(xiàn)变勇,通過(guò)...
    翻炒吧蛋滾飯閱讀 2,316評(píng)論 0 19
  • 0525晨讀感悟 東青 我希望我能懂很多很多的知識(shí)贰锁,能去很遠(yuǎn)很遠(yuǎn)的地方赃梧,能和很牛很牛的人交朋友...... 然而...
    東青喵閱讀 342評(píng)論 3 4