Run Loop 基本概念
Run Loop
就是一個(gè)在線程(thread)里不停執(zhí)行的do-while循環(huán)改备。當(dāng)線程接收到事件(event)時(shí)铺根,Run Loop 內(nèi)的事件處理會(huì)使用對(duì)應(yīng)的句柄(handler)處理事件廊勃。
Run Loop 接受的事件可分為兩種不同的源(source),Input source
傳遞異步事件证舟,通常是其他線程或應(yīng)用發(fā)送過(guò)來(lái)的消息(message)酗洒。Timer sources
傳遞同步事件,即發(fā)生于特定時(shí)間的或以一定時(shí)間間隔循環(huán)發(fā)送的事件凛忿。
上圖展示了 Run Loop 的工作原理:Run Loop 運(yùn)行與線程之中澈灼,從Input source
和Timer sources
接受事件,然后調(diào)用相應(yīng)的 handler 處理事件店溢。iOS框架 Foundation 中定義了 Run Loop 的實(shí)現(xiàn)類NSRunLoop
叁熔。
Run Loop 與線程的關(guān)系
Run Loop 與線程是一一對(duì)應(yīng)的關(guān)系。每一個(gè)線程都有且僅有一個(gè) Run Loop 與其對(duì)應(yīng)床牧,沒(méi)有線程荣回,就沒(méi)有 Run Loop。在iOS應(yīng)用中戈咳,主線程的 Run Loop 是默認(rèn)啟動(dòng)的心软,而其他線程的 Run Loop 默認(rèn)是不啟動(dòng)的革砸。蘋果為我們提供了兩種獲取 Run Loop 對(duì)象的方式:
使用
[NSRunLoop currentRunLoop]
獲取NSRunLoop
對(duì)象使用
CFRunLoopGetCurrent
函數(shù)獲取CFRunLoopRef
獲取的 Run Loop 對(duì)象的線程安全性取決于你所使用的API。Core Foundation 中的函數(shù)通常是線程安全的糯累,可以從任何線程調(diào)用。但是册踩,如果你正在執(zhí)行修改 Run Loop 配置的操作泳姐,那么最佳實(shí)踐是盡可能在 Run Loop 所在的線程進(jìn)行這些操作。
NSRunLoop
類不具有線程安裝性暂吉。如果你使用NSRunLoop
類來(lái)修改 Run Loop胖秒,則應(yīng)僅從持有該 Run Loop 的線程內(nèi)執(zhí)行操作。
Run Loop 的組成部分
一個(gè) Run Loop 包含多個(gè) Mode慕的,每個(gè) Mode 包含多個(gè) Sources阎肝、Objservers 和 Timers。每次調(diào)用 Run Loop 時(shí)肮街,需要指定一種 Mode风题,此時(shí) Run Loop 只能處理該 Mode 包含的Sources、Objservers 和 Timers[1]嫉父。
蘋果官方文檔中提到的 Mode 類型有五種:
iOS中可供調(diào)用的只有NSDefaultRunLoopMode
和NSRunLoopCommonModes
兩個(gè)沛硅,其中NSRunLoopCommonModes
是一個(gè)集合,其中默認(rèn)包括NSDefaultRunLoopMode
和NSEventTrackingRunLoopMode
绕辖。
何時(shí)使用 Run Loop摇肌?
蘋果官方文檔[2]指出,需要顯式運(yùn)行 Run Loop 的唯一時(shí)機(jī)是為應(yīng)用程序創(chuàng)建輔助線程(secondary thread)時(shí)仪际。例如围小,如果你需要執(zhí)行以下任何操作,則需要啟動(dòng) Run Loop:
使用端口(mach port)或自定義輸入源(custom input source)與其他線程通信树碱。
在線程中使用計(jì)時(shí)器(timers)肯适。
使用任何
performSelector
方法。保持線程以執(zhí)行周期任務(wù)赴恨。
其中常用的是timers和performSelector
疹娶。
Timer
計(jì)時(shí)器源(timer source)在將來(lái)的預(yù)設(shè)時(shí)間將事件同步傳遞給線程。蘋果為我們提供了兩種計(jì)時(shí)器的實(shí)現(xiàn)伦连,NSTimer
和CFRunLoopTimerRef
雨饺。
RunLoop為了節(jié)省資源,并不會(huì)在非常準(zhǔn)確的時(shí)間點(diǎn)回調(diào)這個(gè)Timer惑淳。Timer 有個(gè)屬性叫做 Tolerance (寬容度)额港,標(biāo)示了當(dāng)時(shí)間點(diǎn)到后,容許有多少最大誤差歧焦。如果某個(gè)時(shí)間點(diǎn)被錯(cuò)過(guò)了移斩,例如執(zhí)行了一個(gè)很長(zhǎng)的任務(wù)肚医,則那個(gè)時(shí)間點(diǎn)的回調(diào)也會(huì)跳過(guò)去,不會(huì)延后執(zhí)行向瓷。就比如等公交肠套,如果 10:10 時(shí)我忙著玩手機(jī)錯(cuò)過(guò)了那個(gè)點(diǎn)的公交,那我只能等 10:20 這一趟了猖任。[2]
performSelector:
當(dāng)調(diào)用NSObject
的performSelecter:afterDelay:
后你稚,實(shí)際上其內(nèi)部會(huì)創(chuàng)建一個(gè) Timer 并添加到當(dāng)前線程的 RunLoop 中。所以如果當(dāng)前線程沒(méi)有 RunLoop朱躺,則這個(gè)方法會(huì)失效刁赖。當(dāng)調(diào)用performSelector:onThread:
時(shí),實(shí)際上其會(huì)創(chuàng)建一個(gè) Timer 加到對(duì)應(yīng)的線程去长搀,同樣的宇弛,如果對(duì)應(yīng)線程沒(méi)有 Run Loop 該方法也會(huì)失效。[2]