NSRunLoop核心內(nèi)容很多报亩,這里僅結(jié)合自己實(shí)際開(kāi)發(fā)遇到的情況,參考相關(guān)博客岂却,給出自己的一點(diǎn)理解忿薇。主要是NSTimer和線程
一、NSRunLoop是什么鬼躏哩?
NSRunLoop是iOS中的消息處理機(jī)制署浩。一般情況下,一個(gè)線程執(zhí)行完成之后就會(huì)退出扫尺,比如說(shuō)一條NSLog打印語(yǔ)句瑰抵,但是有時(shí)候我們需要一種機(jī)制,執(zhí)行完某個(gè)事件后線程不退出器联,而是進(jìn)入休眠狀態(tài)二汛,當(dāng)再次監(jiān)測(cè)到需要出發(fā)事件時(shí),線程激活拨拓,處理事件肴颊,處理完成后再次進(jìn)入休眠。NSRunLoop就是這樣一種機(jī)制渣磷。像Button的touch事件婿着,UIImageView的gesture事件均是NSRunLoop在起作用。
二、為什么我沒(méi)有覺(jué)察到NSRunLoop的存在
默認(rèn)情況下竟宋,我們并不需要手動(dòng)創(chuàng)建一個(gè)RunLoop(我們也確實(shí)這么做的提完,似乎程序也能正常運(yùn)行),那是因?yàn)閏ocoa框架為我們創(chuàng)建了一個(gè)默認(rèn)的RunLoop丘侠,所以Button的點(diǎn)擊事件才能夠正常執(zhí)行徒欣。
三、NSTimer
一般情況下蜗字,APP首頁(yè)都會(huì)有一個(gè)廣告輪播的效果圖打肝,然后還會(huì)加上定時(shí)器,每隔幾秒切換圖片挪捕。
NSTimer *timer = [NSTimerscheduledTimerWithTimeInterval:5.0f target:selfselector:@selector(timeAction) userInfo:nil repeats:YES];
但是我們發(fā)現(xiàn)粗梭,當(dāng)我們下拉刷新或者滾動(dòng)scrollView(tableView和collection一樣)時(shí),定時(shí)器不工作了级零。為什么呢断医?
1.正如上面所說(shuō),程序啟動(dòng)時(shí)奏纪,系統(tǒng)已經(jīng)在主線程中默認(rèn)加入了RunLoop鉴嗤,這保證了我們的主線程在運(yùn)行起來(lái)后處于一種“等待”的狀態(tài)。而主線程的RunLoop有兩個(gè)Mode:kCFRunLoopDefaultMode(APP平時(shí)所處的狀態(tài)亥贸,幾乎包括所有的輸入源)和UITrackingRunLoopMode(追蹤模式躬窜,當(dāng)滑動(dòng),拖拽時(shí)進(jìn)入的模式)
2.創(chuàng)建一個(gè)Timer炕置,默認(rèn)是加入到RunLoop的DefaultMode荣挨,不滑動(dòng)scrollView時(shí),因?yàn)锳pp默認(rèn)也是在DefaultMode朴摊,所以定時(shí)器正常工作默垄;但是,當(dāng)你滑動(dòng)scrollView的時(shí)候甚纲,RunLoop會(huì)將mode切換為UITrackingRunLoopMode口锭,這時(shí)候Timer就不會(huì)被回調(diào)。
自己理解:RunLoop就是一個(gè)大的循環(huán)圈介杆,會(huì)對(duì)這個(gè)大的循環(huán)圈內(nèi)的相同Mode的事件進(jìn)行檢測(cè)鹃操,當(dāng)監(jiān)聽(tīng)到事件需要處理時(shí),跳轉(zhuǎn)到事件進(jìn)行處理春哨。每次運(yùn)行一個(gè)run loop荆隘,你指定(顯式或隱式)run loop的運(yùn)行模式。當(dāng)相應(yīng)的模式傳遞給run loop時(shí)赴背,只有與該模式對(duì)應(yīng)的input sources才被監(jiān)控并允許run loop對(duì)事件進(jìn)行處理(與此類似椰拒,也只有與該模式對(duì)應(yīng)的observers才會(huì)被通知)
在開(kāi)啟一個(gè)NSTimer實(shí)質(zhì)上是在當(dāng)前的runloop中注冊(cè)了一個(gè)新的事件源晶渠,而當(dāng)scrollView滾動(dòng)的時(shí)候,當(dāng)前的MainRunLoop是處于UITrackingRunLoopMode的模式下燃观,在這個(gè)模式下褒脯,是不會(huì)處理NSDefaultRunLoopMode的消息(因?yàn)镽unLoop Mode不一樣),要想在scrollView滾動(dòng)的同時(shí)也接受其它runloop的消息缆毁,我們需要改變兩者之間的runloopmode.
簡(jiǎn)單的說(shuō)就是NSTimer不會(huì)開(kāi)啟新的進(jìn)程番川,只是在Runloop里注冊(cè)了一下,Runloop每次loop時(shí)都會(huì)檢測(cè)這個(gè)timer积锅,看是否可以觸發(fā)爽彤。當(dāng)Runloop在A mode养盗,而timer注冊(cè)在B mode時(shí)就無(wú)法去檢測(cè)這個(gè)timer缚陷,所以需要把NSTimer也注冊(cè)到A mode,這樣就可以被檢測(cè)到往核。
這里有個(gè)概念叫 "CommonModes":一個(gè) Mode 可以將自己標(biāo)記為"Common"屬性(通過(guò)將其 ModeName 添加到 RunLoop 的 "commonModes" 中)箫爷。每當(dāng) RunLoop 的內(nèi)容發(fā)生變化時(shí),RunLoop 都會(huì)自動(dòng)將 _commonModeItems 里的 Source/Observer/Timer 同步到具有 "Common" 標(biāo)記的所有Mode里聂儒。
應(yīng)用場(chǎng)景舉例:主線程的 RunLoop 里有兩個(gè)預(yù)置的 Mode:kCFRunLoopDefaultMode 和 UITrackingRunLoopMode虎锚。這兩個(gè) Mode 都已經(jīng)被標(biāo)記為"Common"屬性。DefaultMode 是 App 平時(shí)所處的狀態(tài)衩婚,TrackingRunLoopMode 是追蹤 ScrollView 滑動(dòng)時(shí)的狀態(tài)窜护。當(dāng)你創(chuàng)建一個(gè) Timer 并加到 DefaultMode 時(shí),Timer 會(huì)得到重復(fù)回調(diào)非春,但此時(shí)滑動(dòng)一個(gè)TableView時(shí)柱徙,RunLoop 會(huì)將 mode 切換為 TrackingRunLoopMode,這時(shí) Timer 就不會(huì)被回調(diào)奇昙,并且也不會(huì)影響到滑動(dòng)操作护侮。
有時(shí)你需要一個(gè) Timer,在兩個(gè) Mode 中都能得到回調(diào)储耐,一種辦法就是將這個(gè) Timer 分別加入這兩個(gè) Mode羊初。還有一種方式,就是將 Timer 加入到頂層的 RunLoop 的 "commonModeItems" 中什湘。"commonModeItems" 被 RunLoop 自動(dòng)更新到所有具有"Common"屬性的 Mode 里去长赞。
[[NSRunLoop currentRunLoop] addTimer:timer forMode: NSRunLoopCommonModes];//加入這句即可
NSRunLoopCommonModes 這是一組可配置的通用模式。將input sources與該模式關(guān)聯(lián)則同時(shí)也將input sources與該組中的其它模式進(jìn)行了關(guān)聯(lián)闽撤。
這里使用的模式是:NSRunLoopCommonModes得哆,這個(gè)模式等效于NSDefaultRunLoopMode和NSEventTrackingRunLoopMode的結(jié)合。