學(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è)
RunLoopModle中sources為空、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.