RunLoop-基礎(chǔ)概念(初識(shí)篇)

學(xué)習(xí)這篇內(nèi)容主要講解RunLoop的概念,以及RunLoop和線程之間的關(guān)系。
當(dāng)然提及RunLoop也離不開(kāi)Autorealse Pool,本篇內(nèi)容略有提及,但不重點(diǎn)闡述。
本篇內(nèi)容是我自己對(duì)RunLoop概念的總結(jié),和簡(jiǎn)單呈現(xiàn),內(nèi)容比較精煉。

概念
  • RunLoop是系統(tǒng)中和線程相關(guān)的基礎(chǔ)架構(gòu)的組成部分,一個(gè)RunLoop是一個(gè)事件處理環(huán),系統(tǒng)利用這個(gè)事件處理環(huán)來(lái)安排事務(wù)。
  • RunLoop的意義是讓你的線程在有工作的時(shí)候去干活,沒(méi)有工作的時(shí)候進(jìn)入休眠節(jié)省系統(tǒng)資源。
  • 每個(gè)線程(包含主線程)都有一個(gè)Runloop。對(duì)于每一個(gè)Runloop,系統(tǒng)會(huì)隱式創(chuàng)建一個(gè)Autorelease pool,這樣所有的Autorelease Pool會(huì)構(gòu)成一個(gè)像callstack一樣的一個(gè)棧式結(jié)構(gòu),在每一個(gè)Runloop結(jié)束時(shí),當(dāng)前棧頂?shù)腁utorelease pool會(huì)被銷毀,這樣這個(gè)Pool里的每個(gè)Object會(huì)被release。
  • RunLoop和線程之間是以鍵值對(duì)應(yīng)的形式一一對(duì)應(yīng)的,其中key是thread,value是RunLoop,RunLoop也是管理線程的一種機(jī)制,這種機(jī)制不僅在iOS上有,在安卓的Looper,Node.js中的EventLoop都有類似的模式。
    喚醒一個(gè)線程其實(shí)就是喚醒線程的source。

一、初識(shí)RunLoop

首先,我們?cè)谥骶€程中添加如下代碼:

while (1) {
    NSLog(@"while begin");
    // the thread be blocked here
    NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
    [runLoop runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
    // this will not be executed
    NSLog(@"while end");
}

我們將上面代碼在主線程運(yùn)行,我們會(huì)發(fā)現(xiàn)while end沒(méi)有執(zhí)行,過(guò)一會(huì)又執(zhí)行了,這是因?yàn)椋?/p>

  • 主線程中,本身有自己的RunLoop,所以主線程可以一直不被釋放,在需要做事情的時(shí)候主線程被喚醒干活,
  • 在不需要做事情的時(shí)候主線程會(huì)休眠,所以上面代碼到distantFuture休眠線程后,會(huì)停止執(zhí)行,當(dāng)主線程需要處理某些事情的時(shí)候才會(huì)被喚醒。
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    
    while (1) {
        NSLog(@"while begin");
        NSRunLoop *subRunLoop = [NSRunLoop currentRunLoop];
        [subRunLoop runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
        NSLog(@"while end");
    }
});

上面的代碼是通過(guò)GCD開(kāi)啟全局的一個(gè)子線程,運(yùn)行代碼后,在子線程會(huì)無(wú)限循環(huán)的一直在跑,不會(huì)停!這是因?yàn)?

  • 這個(gè)RunLoopModlesources為空、observers為空、timers為空,所以這個(gè)RunLoop直接就結(jié)束釋放了.
  • 我們看到雖然有Mode,但是我們沒(méi)有給它soures,observer,timer,其實(shí)Mode中的這些source,observer,timer,統(tǒng)稱為這個(gè)Mode的item,如果一個(gè)Mode中一個(gè)item都沒(méi)有,則這個(gè)RunLoop會(huì)直接退出,不進(jìn)入循環(huán)(其實(shí)線程之所以可以一直存在就是由于RunLoop將其帶入了這個(gè)循環(huán)中)。下面我們?yōu)檫@個(gè)RunLoop添加個(gè)source:
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    
    while (1) {
        NSPort *macPort = [NSPort port];
        NSLog(@"while begin");
        NSRunLoop *subRunLoop = [NSRunLoop currentRunLoop];
        [subRunLoop addPort:macPort forMode:NSDefaultRunLoopMode];
        [subRunLoop runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
        NSLog(@"while end");
        NSLog(@"%@",subRunLoop);
        
    }
});

上面的代碼,運(yùn)行后,會(huì)停在休眠的那一行代碼,因?yàn)槲覀兘oRunLoop的model添加item.

小結(jié):我們的RunLoop要想工作,必須要讓它存在一個(gè)Item(source,observer或者timer),主線程之所以能夠一直存在,并且隨時(shí)準(zhǔn)備被喚醒就是應(yīng)為系統(tǒng)為其添加了很多Item.

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

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

  • 轉(zhuǎn)自bireme,原地址:https://blog.ibireme.com/2015/05/18/runloop/...
    乜_啊_閱讀 1,675評(píng)論 0 5
  • 轉(zhuǎn)自http://blog.ibireme.com/2015/05/18/runloop 深入理解RunLoop ...
    飄金閱讀 1,065評(píng)論 0 4
  • 原文地址:http://blog.ibireme.com/2015/05/18/runloop/ RunLoop ...
    大餅炒雞蛋閱讀 1,259評(píng)論 0 6
  • RunLoop 的概念 一般來(lái)講,一個(gè)線程一次只能執(zhí)行一個(gè)任務(wù),執(zhí)行完成后線程就會(huì)退出。如果我們需要一個(gè)機(jī)制,讓線...
    Mirsiter_魏閱讀 670評(píng)論 0 2
  • 二零一九年二月十四日,情人節(jié),我想你了。
    傘傘傘兒閱讀 126評(píng)論 0 1

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