學習這篇內(nèi)容主要講解RunLoop的概念,以及RunLoop和線程之間的關(guān)系罗晕。
當然提及RunLoop也離不開Autorealse Pool,本篇內(nèi)容略有提及仔掸,但不重點闡述。
本篇內(nèi)容是我自己對RunLoop概念的總結(jié),和簡單呈現(xiàn)量蕊,內(nèi)容比較精煉铺罢。
概念
- RunLoop是系統(tǒng)中和線程相關(guān)的基礎(chǔ)架構(gòu)的組成部分,一個RunLoop是一個事件處理環(huán)残炮,系統(tǒng)利用這個事件處理環(huán)來安排事務(wù)韭赘。
- RunLoop的意義是讓你的線程在有工作的時候去干活势就,沒有工作的時候進入休眠節(jié)省系統(tǒng)資源。
- 每個線程(包含主線程)都有一個Runloop苞冯。對于每一個Runloop,系統(tǒng)會隱式創(chuàng)建一個Autorelease pool舅锄,這樣所有的Autorelease Pool會構(gòu)成一個像callstack一樣的一個棧式結(jié)構(gòu)鞭达,在每一個Runloop結(jié)束時,當前棧頂?shù)腁utorelease pool會被銷毀皇忿,這樣這個Pool里的每個Object會被release畴蹭。
- RunLoop和線程之間是以鍵值對應(yīng)的形式一一對應(yīng)的,其中key是thread禁添,value是RunLoop撮胧,RunLoop也是管理線程的一種機制,這種機制不僅在iOS上有老翘,在安卓的Looper芹啥,Node.js中的EventLoop都有類似的模式。
喚醒一個線程其實就是喚醒線程的source铺峭。
一墓怀、初識RunLoop
首先,我們在主線程中添加如下代碼:
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");
}
我們將上面代碼在主線程運行卫键,我們會發(fā)現(xiàn)while end
沒有執(zhí)行傀履,過一會又執(zhí)行了,這是因為:
- 主線程中莉炉,本身有自己的
RunLoop
钓账,所以主線程可以一直不被釋放,在需要做事情的時候主線程被喚醒干活絮宁, - 在不需要做事情的時候主線程會休眠梆暮,所以上面代碼到
distantFuture
休眠線程后,會停止執(zhí)行绍昂,當主線程需要處理某些事情的時候才會被喚醒啦粹。
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");
}
});
上面的代碼是通過GCD
開啟全局的一個子線程偿荷,運行代碼后,在子線程會無限循環(huán)的一直在跑唠椭,不會停跳纳!這是因為:
- 這個
RunLoopModle
中sources
為空、observers
為空贪嫂、timers
為空寺庄,所以這個RunLoop
直接就結(jié)束釋放了. - 我們看到雖然有Mode,但是我們沒有給它
soures,observer,timer
撩荣,其實Mode中的這些source,observer,timer
铣揉,統(tǒng)稱為這個Mode的item,如果一個Mode中一個item都沒有餐曹,則這個RunLoop會直接退出,不進入循環(huán)(其實線程之所以可以一直存在就是由于RunLoop
將其帶入了這個循環(huán)中)敌厘。下面我們?yōu)檫@個RunLoop
添加個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);
}
});
上面的代碼台猴,運行后,會停在休眠的那一行代碼俱两,因為我們給RunLoop的model添加item.
小結(jié):我們的RunLoop
要想工作饱狂,必須要讓它存在一個Item(source,observer或者timer
),主線程之所以能夠一直存在宪彩,并且隨時準備被喚醒就是應(yīng)為系統(tǒng)為其添加了很多Item.