iOS開發-Runloop詳解(簡書)

不知道你們有沒有想過這個問題,一個應用開始運行之後放在那裏,若是不對它進行任何操做,這個應用就像靜止了同樣,不會自發的有任何動做發生,可是若是咱們點擊界面上的一個按鈕,這個時候就會有對應的按鈕響應事件發生。給咱們的感受就像應用一直處於隨時待命的狀態,在沒人操做的時候它一直在休息,在讓它幹活的時候,它就能馬上響應。其實,這就是run loop的功勞。編程

1、線程與run loopxcode

1.1 線程任務的類型安全

再來講說線程。有些線程執行的任務是一條直線,起點到終點;而另外一些線程要乾的活則是一個圓,不斷循環,直到經過某種方式將它終止。直線線程如簡單的Hello World,運行打印完,它的生命週期便結束了,像曇花一現那樣;圓類型的如操做系統,一直運行直到你關機。在IOS中,圓型的線程就是經過run loop不停的循環實現的。架構

1.2 線程與run loop的關係app

Run loop,正如其名,loop表示某種循環,和run放在一塊兒就表示一直在運行着的循環。實際上,run loop和線程是緊密相連的,能夠這樣說run loop是爲了線程而生,沒有線程,它就沒有存在的必要。Run loops是線程的基礎架構部分,Cocoa和CoreFundation都提供了run loop對象方便配置和管理線程的run loop(如下都已Cocoa爲例)。每一個線程,包括程序的主線程(main thread)都有與之相應的run loop對象。異步

1.2.1 主線程的run loop默認是啓動的。函數

iOS的應用程序裏面,程序啓動後會有一個以下的main() 函數:oop

     int main(int argc, char *argv[])操作系統

     {線程

            @autoreleasepool {

              return UIApplicationMain(argc, argv, nil, NSStringFromClass([appDelegate class]));

           }

  }

重點是UIApplicationMain() 函數,這個方法會爲main thread 設置一個NSRunLoop 對象,這就解釋了本文開始說的爲何咱們的應用能夠在無人操做的時候休息,須要讓它幹活的時候又能立馬響應。

1.2.2 對其它線程來講,run loop默認是沒有啓動的,若是你須要更多的線程交互則能夠手動配置和啓動,若是線程只是去執行一個長時間的已肯定的任務則不須要。

1.2.3 在任何一個Cocoa程序的線程中,均可以經過:

NSRunLoop   *runloop = [NSRunLoop currentRunLoop];

來獲取到當前線程的run loop。

1.3 關於run loop的幾點說明

1.3.1 Cocoa中的NSRunLoop類並非線程安全的

咱們不能再一個線程中去操做另一個線程的run loop對象,那極可能會形成意想不到的後果。不過幸運的是CoreFundation中的不透明類CFRunLoopRef是線程安全的,並且兩種類型的run loop徹底能夠混合使用。Cocoa中的NSRunLoop類能夠經過實例方法:

- (CFRunLoopRef)getCFRunLoop;

獲取對應的CFRunLoopRef類,來達到線程安全的目的。

1.3.2 Run loop的管理並不徹底是自動的。

咱們仍必須設計線程代碼以在適當的時候啓動run loop並正確響應輸入事件,固然前提是線程中須要用到run loop。並且,咱們還須要使用while/for語句來驅動run loop可以循環運行,下面的代碼就成功驅動了一個run loop:

     BOOL isRunning = NO;

      do {

            isRunning = [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDatedistantFuture]];

     } while (isRunning);

1.3.3 Run loop同時也負責autorelease pool的建立和釋放

在使用手動的內存管理方式的項目中,會常常用到不少自動釋放的對象,若是這些對象不可以被即時釋放掉,會形成內存佔用量急劇增大。Run loop就爲咱們作了這樣的工做,每當一個運行循環結束的時候,它都會釋放一次autorelease pool,同時pool中的全部自動釋放類型變量都會被釋放掉。

1.3.4 Run loop的優勢

一個run loop就是一個事件處理循環,用來不停的監聽和處理輸入事件並將其分配到對應的目標上進行處理。若是僅僅是想實現這個功能,你可能會想一個簡單的while循環不就能夠實現了嗎,用得着費老大勁來作個那麼複雜的機制?顯然,蘋果的架構設計師不是吃乾飯的,你想到的他們早就想過了。

首先,NSRunLoop是一種更加高明的消息處理模式,他就高明在對消息處理過程進行了更好的抽象和封裝,這樣才能是的你不用處理一些很瑣碎很低層次的具體消息的處理,在NSRunLoop中每個消息就被打包在input source或者是timer source(見後文)中了。

其次,也是很重要的一點,使用run loop可使你的線程在有工做的時候工做,沒有工做的時候休眠,這能夠大大節省系統資源。

 

2、Run loop相關知識點

2.1輸入事件來源

Run loop接收輸入事件來自兩種不一樣的來源:輸入源(input source)和定時源(timer source)。兩種源都使用程序的某一特定的處理例程來處理到達的事件。圖-1顯示了run loop的概念結構以及各類源。


須要說明的是,當你建立輸入源,你須要將其分配給run loop中的一個或多個模式(什麼是模式,下文將會講到)。模式只會在特定事件影響監聽的源。大多數狀況下,run loop運行在默認模式下,可是你也可使其運行在自定義模式。若某一源在當前模式下不被監聽,那麼任何其生成的消息只在run loop運行在其關聯的模式下才會被傳遞。

圖-1  Runloop的結構和輸入源類型

 

2.1.1輸入源(input source)

傳遞異步事件,一般消息來自於其餘線程或程序。輸入源傳遞異步消息給相應的處理例程,並調用runUntilDate:方法來退出(在線程裏面相關的NSRunLoop對象調用)。

2.1.1.1基於端口的輸入源

基於端口的輸入源由內核自動發送。

Cocoa和Core Foundation內置支持使用端口相關的對象和函數來建立的基於端口的源。例如,在Cocoa裏面你歷來不須要直接建立輸入源。你只要簡單的建立端口對象,並使用NSPort的方法把該端口添加到run loop。端口對象會本身處理建立和配置輸入源。

在Core Foundation,你必須人工建立端口和它的run loop源。咱們可使用端口相關的函數(CFMachPortRef,CFMessagePortRef,CFSocketRef)來建立合適的對象。下面的例子展現瞭如何建立一個基於端口的輸入源,將其添加到run loop並啓動:

voidcreatePortSource()

{

    CFMessagePortRef port = CFMessagePortCreateLocal(kCFAllocatorDefault, CFSTR("com.someport"),myCallbackFunc, NULL, NULL);

    CFRunLoopSourceRef source =  CFMessagePortCreateRunLoopSource(kCFAllocatorDefault, port, 0);

    CFRunLoopAddSource(CFRunLoopGetCurrent(), source, kCFRunLoopCommonModes);

    while (pageStillLoading) {

        NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];

        CFRunLoopRun();

        [pool release];

    }

    CFRunLoopRemoveSource(CFRunLoopGetCurrent(), source, kCFRunLoopDefaultMode);

    CFRelease(source);

}

2.1.1.2自定義輸入源

自定義的輸入源須要人工從其餘線程發送。

爲了建立自定義輸入源,必須使用Core Foundation裏面的CFRunLoopSourceRef類型相關的函數來建立。你可使用回調函數來配置自定義輸入源。Core Fundation會在配置源的不一樣地方調用回調函數,處理輸入事件,在源從run loop移除的時候清理它。

除了定義在事件到達時自定義輸入源的行爲,你也必須定義消息傳遞機制。源的這部分運行在單獨的線程裏面,並負責在數據等待處理的時候傳遞數據給源並通知它處理數據。消息傳遞機制的定義取決於你,但最好不要過於複雜。建立並啓動自定義輸入源的示例以下:

voidcreateCustomSource()

{

    CFRunLoopSourceContext context = {0, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL};

    CFRunLoopSourceRef source = CFRunLoopSourceCreate(kCFAllocatorDefault, 0, &context);

    CFRunLoopAddSource(CFRunLoopGetCurrent(), source, kCFRunLoopDefaultMode);

    while (pageStillLoading) {

        NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];

        CFRunLoopRun();

        [pool release];

    }

    CFRunLoopRemoveSource(CFRunLoopGetCurrent(), source, kCFRunLoopDefaultMode);

    CFRelease(source);

}

2.1.1.3Cocoa上的Selector源

除了基於端口的源,Cocoa定義了自定義輸入源,容許你在任何線程執行selector方法。和基於端口的源同樣,執行selector請求會在目標線程上序列化,減緩許多在線程上容許多個方法容易引發的同步問題。不像基於端口的源,一個selector執行完後會自動從run loop裏面移除。

當在其餘線程上面執行selector時,目標線程須有一個活動的run loop。對於你建立的線程,這意味着線程在你顯式的啓動run loop以前是不會執行selector方法的,而是一直處於休眠狀態。

NSObject類提供了相似以下的selector方法:

- (void)performSelectorOnMainThread:(SEL)aSelector withObject:(id)argwaitUntilDone:(BOOL)wait modes:(NSArray *)array;

 

2.1.2定時源(timer source)

定時源在預設的時間點同步方式傳遞消息,這些消息都會發生在特定時間或者重複的時間間隔。定時源則直接傳遞消息給處理例程,不會當即退出run loop。

須要注意的是,儘管定時器能夠產生基於時間的通知,但它並非實時機制。和輸入源同樣,定時器也和你的run loop的特定模式相關。若是定時器所在的模式當前未被run loop監視,那麼定時器將不會開始直到run loop運行在相應的模式下。相似的,若是定時器在run loop處理某一事件期間開始,定時器會一直等待直到下次run loop開始相應的處理程序。若是run loop再也不運行,那定時器也將永遠不啓動。

建立定時器源有兩種方法,

方法一:

NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:4.0

                                                     target:self

                                                   selector:@selector(backgroundThreadFire:) userInfo:nil

                                                    repeats:YES];

    [[NSRunLoop currentRunLoop] addTimer:timerforMode:NSDefaultRunLoopMode];

 

方法二:

[NSTimer scheduledTimerWithTimeInterval:10

                                        target:self

                                       selector:@selector(backgroundThreadFire:)

                                       userInfo:nil

                                       repeats:YES];

2.2 RunLoop觀察者

源是在合適的同步或異步事件發生時觸發,而run loop觀察者則是在run loop自己運行的特定時候觸發。你可使用run loop觀察者來爲處理某一特定事件或是進入休眠的線程作準備。你能夠將run loop觀察者和如下事件關聯:

1.  Runloop入口

2.  Runloop什麼時候處理一個定時器

3.  Runloop什麼時候處理一個輸入源

4.  Runloop什麼時候進入睡眠狀態

5.  Runloop什麼時候被喚醒,但在喚醒以前要處理的事件

6.  Runloop終止

和定時器相似,在建立的時候你能夠指定run loop觀察者能夠只用一次或循環使用。若只用一次,那麼在它啓動後,會把它本身從run loop裏面移除,而循環的觀察者則不會。定義觀察者並把它添加到run loop,只能使用Core Fundation。下面的例子演示瞭如何建立run loop的觀察者:

- (void)addObserverToCurrentRunloop

{

    // The application uses garbage collection, so noautorelease pool is needed.

    NSRunLoop*myRunLoop = [NSRunLoop currentRunLoop];

   

    // Create a run loop observer and attach it to the runloop.

    CFRunLoopObserverContext  context = {0, self, NULL, NULL, NULL};

   CFRunLoopObserverRef    observer =CFRunLoopObserverCreate(kCFAllocatorDefault,

                                                              kCFRunLoopBeforeTimers, YES, 0, &myRunLoopObserver, &context);

   

    if (observer)

    {

        CFRunLoopRef    cfLoop = [myRunLoop getCFRunLoop];

       CFRunLoopAddObserver(cfLoop, observer, kCFRunLoopDefaultMode);

    }

}

其中,kCFRunLoopBeforeTimers表示選擇監聽定時器觸發前處理事件,後面的YES表示循環監聽。

 

2.3 RunLoop的事件隊列

每次運行run loop,你線程的run loop對會自動處理以前未處理的消息,並通知相關的觀察者。具體的順序以下:

  1. 通知觀察者run loop已經啓動
  2. 通知觀察者任何即將要開始的定時器
  3. 通知觀察者任何即將啓動的非基於端口的源
  4. 啓動任何準備好的非基於端口的源
  5. 若是基於端口的源準備好並處於等待狀態,當即啓動;並進入步驟9。
  6. 通知觀察者線程進入休眠
  7. 將線程置於休眠直到任一下面的事件發生:
    • 某一事件到達基於端口的源
    • 定時器啓動
    • Run loop設置的時間已經超時
    • run loop被顯式喚醒
  8. 通知觀察者線程將被喚醒。
  9. 處理未處理的事件
    • 若是用戶定義的定時器啓動,處理定時器事件並重啓run loop。進入步驟2
    • 若是輸入源啓動,傳遞相應的消息
    • 若是run loop被顯式喚醒並且時間還沒超時,重啓run loop。進入步驟2
  10. 通知觀察者run loop結束。

由於定時器和輸入源的觀察者是在相應的事件發生以前傳遞消息,因此通知的時間和實際事件發生的時間之間可能存在偏差。若是須要精確時間控制,你可使用休眠和喚醒通知來幫助你校對實際發生事件的時間。

由於當你運行run loop時定時器和其它週期性事件常常須要被傳遞,撤銷run loop也會終止消息傳遞。典型的例子就是鼠標路徑追蹤。由於你的代碼直接獲取到消息而不是經由程序傳遞,所以活躍的定時器不會開始直到鼠標追蹤結束並將控制權交給程序。

Run loop能夠由run loop對象顯式喚醒。其它消息也能夠喚醒run loop。例如,添加新的非基於端口的源會喚醒run loop從而能夠當即處理輸入源而不須要等待其餘事件發生後再處理。

從這個事件隊列中能夠看出:

①若是是事件到達,消息會被傳遞給相應的處理程序來處理, runloop處理完當次事件後,run loop會退出,而無論以前預約的時間到了沒有。你能夠從新啓動run loop來等待下一事件。

②若是線程中有須要處理的源,可是響應的事件沒有到來的時候,線程就會休眠等待相應事件的發生。這就是爲何run loop能夠作到讓線程有工做的時候忙於工做,而沒工做的時候處於休眠狀態。

 

2.4何時使用run loop

僅當在爲你的程序建立輔助線程的時候,你才須要顯式運行一個run loop。Run loop是程序主線程基礎設施的關鍵部分。因此,Cocoa和Carbon程序提供了代碼運行主程序的循環並自動啓動run loop。IOS程序中UIApplication的run方法(或Mac OS X中的NSApplication)做爲程序啓動步驟的一部分,它在程序正常啓動的時候就會啓動程序的主循環。相似的,RunApplicationEventLoop函數爲Carbon程序啓動主循環。若是你使用xcode提供的模板建立你的程序,那你永遠不須要本身去顯式的調用這些例程。

對於輔助線程,你須要判斷一個run loop是不是必須的。若是是必須的,那麼你要本身配置並啓動它。你不須要在任何狀況下都去啓動一個線程的run loop。好比,你使用線程來處理一個預先定義的長時間運行的任務時,你應該避免啓動run loop。Run loop在你要和線程有更多的交互時才須要,好比如下狀況:

  1. 使用端口或自定義輸入源來和其餘線程通訊
  2. 使用線程的定時器
  3. Cocoa中使用任何performSelector…的方法
  4. 使線程週期性工做

若是你決定在程序中使用run loop,那麼它的配置和啓動都很簡單。和全部線程編程同樣,你須要計劃好在輔助線程退出線程的情形。讓線程天然退出每每比強制關閉它更好。

另外.....

個人願望是.......

世界和平.........

相關文章
相關標籤/搜索