[ios 程序啓動與運轉] - RunLoop我的小結

 

學習iOS開發通常都是從UI開始的,從只知道從IB拖控件,到知道怎麼在方法裏寫代碼,而後會顯示什麼樣的視圖,產生什麼樣的事件,等等。其實程序從啓動開始,一直都是按照蘋果封裝好的代碼運行着,暴露的一些屬性和方法做爲接口,是讓咱們在給定的方法裏寫代碼實現自定義功能,作出各類各樣的應用。這些方法的調用順序最爲關鍵,熟悉了程序運轉和方法調用的順序,才能夠更好地操控程序和代碼,儘可能避免Xcode不報錯又實現不了功能的BUG。從Xcode的線程函數調用棧能夠看到一些方法調用順序。程序員


 
--零--從程序啓動開始到view顯示:

start---->(加載framework,動態靜態連接庫,啓動圖片,Info.plist,pch等)---->main函數---->UIApplicationMain函數:安全

- 初始化UIApplication單例對象 - 初始化AppDelegate對象,並設爲UIApplication對象的代理 - 檢查Info.plist設置的xib文件是否有效,若是有則解凍Nib文件並設置outlets,建立顯示key window、rootViewController、與rootViewController關聯的根view(沒有關聯則看rootViewController同名的xib),不然launch以後由程序員手動加載。 - 創建一個主事件循環,其中包含UIApplication的Runloop來開始處理事件。

UIApplication
  一、經過window管理視圖;
  二、發送Runloop封裝好的control消息給target;
  三、處理URL,應用圖標警告,聯網狀態,狀態欄,遠程事件等。
AppDelegate
管理UIApplication生命週期和應用的五種狀態(notRunning/inactive/active/background/suspend)。
Key Window
  一、顯示view;
  二、管理rootViewcontroller生命週期;
  三、發送UIApplication傳來的事件消息給view。
rootViewController
一、管理view(view生命週期;view的數據源/代理;view與superView之間事件響應nextResponder的「備胎」);
二、界面跳轉與傳值;
三、狀態欄,屏幕旋轉。
view
  一、經過做爲CALayer的代理,管理layer的渲染(順序大概是先更新約束,再layout再display)和動畫(默認layer的屬性可動畫,view默認禁止,在UIView的block分類方法裏纔打開動畫)。layer是RGBA紋理,經過和mask位圖(含alpha屬性)關聯將合成後的layer紋理填充在像素點內,GPU每1/60秒將計算出的紋理display在像素點中。
  二、佈局子控件(屏幕旋轉或者子視圖佈局變更時,view會從新佈局)。
  三、事件響應:event和guesture。
插播控制器生命週期
runloop:
  一、(要讓馬兒跑)經過do-while死循環讓程序持續運行:接收用戶輸入,調度處理事件時間。
  二、(要讓馬兒少吃草)經過mach_msg()讓runloop沒事時進入trap狀態,節省CPU資源。網絡


  關於程序啓動原理以及各個控件的資料,已經有太多資料介紹,平時咱們也常常接觸常常用到,但關於Runloop的資料,官方文檔老是太過簡練,網上資源說法也不太統一,只能從CFRunLoopRef開源代碼着手,試着學習總結下。(NSRunloop是對CFRunloopRef的面向對象封裝,可是不是線程安全)。數據結構


 

--一--Runloop

 

一、與線程和自動釋放池相關: 二、CFRunLoopRef構造:數據結構;建立與退出;mode切換和item依賴;Runloop啓動 - CFRunLoopModeRef:數據結構(與CFRunLoopRef放一塊兒了);建立;類型; modeItems:- CFRunLoopSourceRef:數據結構(source0/source1); - source0 : - source1 : - CFRunLoopTimerRef:數據結構;建立與生效;相關類型(GCD的timer與CADisplayLink) - CFRunLoopObserverRef:數據結構;建立與添加;監聽的狀態; 三、Runloop內部邏輯:關鍵在兩個判斷點(是否睡覺,是否退出) - 代碼實現: - 函數做用棧顯示: 四、Runloop本質:mach port和mach_msg()。 五、如何處理事件: - 界面刷新: - 手勢識別: - GCD任務: - timer:(與CADisplayLink) - 網絡請求: 六、應用: - 滑動與圖片刷新; - 常駐子線程,保持子線程一直處理事件

 

 

Runloop

一、與線程和自動釋放池相關:app

Runloop的寄生於線程:一個線程只能有惟一對應的runloop;但這個根runloop裏能夠嵌套子runloops;
自動釋放池寄生於Runloop:程序啓動後,主線程註冊了兩個Observer監聽runloop的進出與睡覺。一個最高優先級OB監測Entry狀態;一個最低優先級OB監聽BeforeWaiting狀態和Exit狀態。
線程(建立)-->runloop將進入-->最高優先級OB建立釋放池-->runloop將睡-->最低優先級OB銷燬舊池建立新池-->runloop將退出-->最低優先級OB銷燬新池-->線程(銷燬)異步


二、CFRunLoopRef構造:socket

數據結構:async

// runloop數據結構 struct __CFRunLoopMode { CFStringRef _name; // Mode名字, CFMutableSetRef _sources0; // Set<CFRunLoopSourceRef> CFMutableSetRef _sources1; // Set<CFRunLoopSourceRef> CFMutableArrayRef _observers; // Array<CFRunLoopObserverRef> CFMutableArrayRef _timers; // Array<CFRunLoopTimerRef> ... }; // mode數據結構 struct __CFRunLoop { CFMutableSetRef _commonModes; // Set<CFStringRef> CFMutableSetRef _commonModeItems; // Set<Source/Observer/Timer> CFRunLoopModeRef _currentMode; // Current Runloop Mode CFMutableSetRef _modes; // Set<CFRunLoopModeRef> ... };

建立與退出:mode切換和item依賴函數

a 主線程的runloop自動建立,子線程的runloop默認不建立(在子線程中調用NSRunLoop *runloop = [NSRunLoop currentRunLoop]; 獲取RunLoop對象的時候,就會建立RunLoop); b runloop退出的條件:app退出;線程關閉;設置最大時間到期;modeItem爲空; c 同一時間一個runloop只能在一個mode,切換mode只能退出runloop,再重進指定mode(隔離modeItems使之互不干擾);  d 一個item能夠加到不一樣mode;一個mode被標記到commonModes裏(這樣runloop不用切換mode)。

啓動Runloop:oop

// 用DefaultMode啓動 void CFRunLoopRun(void) { CFRunLoopRunSpecific(CFRunLoopGetCurrent(), kCFRunLoopDefaultMode, 1.0e10, false); } // 用指定的Mode啓動,容許設置RunLoop最大時間(假無限循環),執行完畢是否退出 int CFRunLoopRunInMode(CFStringRef modeName, CFTimeInterval seconds, Boolean stopAfterHandle) { return CFRunLoopRunSpecific(CFRunLoopGetCurrent(), modeName, seconds, returnAfterSourceHandled); }
  • CFRunLoopModeRef:
    數據結構(見上);
    建立添加:runloop自動建立對應的mode;mode只能添加不能刪除

    // 添加mode CFRunLoopAddCommonMode(CFRunLoopRef runloop, CFStringRef modeName);

類型:

1. kCFRunLoopDefaultMode: 默認 mode,一般主線程在這個 Mode 下運行。 2. UITrackingRunLoopMode: 追蹤mode,保證Scrollview滑動順暢不受其餘 mode 影響。 3. UIInitializationRunLoopMode: 啓動程序後的過渡mode,啓動完成後就再也不使用。 4: GSEventReceiveRunLoopMode: Graphic相關事件的mode,一般用不到。 5: kCFRunLoopCommonModes: 佔位mode,做爲標記DefaultMode和CommonMode用。
  • modeItems:

// 添加移除item的函數(參數:添加/移除哪一個item到哪一個runloop的哪一個mode下) CFRunLoopAddSource(CFRunLoopRef rl, CFRunLoopSourceRef source, CFStringRef modeName); CFRunLoopAddObserver(CFRunLoopRef rl, CFRunLoopObserverRef observer, CFStringRef modeName); CFRunLoopAddTimer(CFRunLoopRef rl, CFRunLoopTimerRef timer, CFStringRef mode); CFRunLoopRemoveSource(CFRunLoopRef rl, CFRunLoopSourceRef source, CFStringRef modeName);  CFRunLoopRemoveObserver(CFRunLoopRef rl, CFRunLoopObserverRef observer, CFStringRef modeName); CFRunLoopRemoveTimer(CFRunLoopRef rl, CFRunLoopTimerRef timer, CFStringRef mode);

A-- CFRunLoopSourceRef:事件來源

按照官方文檔CFRunLoopSourceRef爲3類,但數據結構只有兩類(???) Port-Based Sources:與內核端口相關 Custom Input Sources:與自定義source相關 Cocoa Perform Selector Sources:與PerformSEL方法相關)

數據結構(source0/source1);

// source0 (manual): order(優先級),callout(回調函數) CFRunLoopSource {order =..., {callout =... }} // source1 (mach port):order(優先級),port:(端口), callout(回調函數) CFRunLoopSource {order = ..., {port = ..., callout =...}

source0:event事件,只含有回調,須要標記待處理(signal),而後手動將runloop喚醒(wakeup);
source1 :包含一個 mach_port 和一個回調,被用於經過內核和其餘線程發送的消息,能主動喚醒runloop。

B-- CFRunLoopTimerRef:系統內「定時鬧鐘」

NSTimer和performSEL方法其實是對CFRunloopTimerRef的封裝;runloop啓動時設置的最大超時時間其實是GCD的dispatch_source_t類型。

數據結構:

// Timer:interval:(鬧鐘間隔), tolerance:(延期時間容忍度),callout(回調函數) CFRunLoopTimer {firing =..., interval = ...,tolerance = ...,next fire date = ...,callout = ...}

建立與生效;

//NSTimer: // 建立一個定時器(須要手動加到runloop的mode中) + (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti invocation:(NSInvocation *)invocation repeats:(BOOL)yesOrNo; // 默認已經添加到主線程的runLoop的DefaultMode中 + (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti invocation:(NSInvocation *)invocation repeats:(BOOL)yesOrNo;  // performSEL方法 // 內部會建立一個Timer到當前線程的runloop中(若是當前線程沒runloop則方法無效;performSelector:onThread: 方法放到指定線程runloop中) - (void)performSelector:(SEL)aSelector withObject:(id)anArgument afterDelay:(NSTimeInterval)delay

相關類型(GCD的timer與CADisplayLink)

GCD的timer:
dispatch_source_t 類型,能夠精確的參數,不用以來runloop和mode,性能消耗更小。

dispatch_source_set_timer(dispatch_source_t source, // 定時器對象 dispatch_time_t start, // 定時器開始執行的時間 uint64_t interval, // 定時器的間隔時間 uint64_t leeway // 定時器的精度 );

CADisplayLink :
Timer的tolerance表示最大延期時間,若是由於阻塞錯過了這個時間精度,這個時間點的回調也會跳過去,不會延後執行。
CADisplayLink 是一個和屏幕刷新率一致的定時器,若是在兩次屏幕刷新之間執行了一個長任務,那其中就會有一幀被跳過去(和 NSTimer 類似,只是沒有tolerance容忍時間),形成界面卡頓的感受。

C--CFRunLoopObserverRef:監聽runloop狀態,接收回調信息(常見於自動釋放池建立銷燬)

數據結構:

// Observer:order(優先級),ativity(監聽狀態),callout(回調函數) CFRunLoopObserver {order = ..., activities = ..., callout = ...}

建立與添加;

// 第一個參數用於分配該observer對象的內存空間 // 第二個參數用以設置該observer監聽什麼狀態 // 第三個參數用於標識該observer是在第一次進入run loop時執行仍是每次進入run loop處理時均執行 // 第四個參數用於設置該observer的優先級,通常爲0 // 第五個參數用於設置該observer的回調函數 // 第六個參數observer的運行狀態 CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) { // 執行代碼 }

監聽的狀態;

typedef CF_OPTIONS(CFOptionFlags, CFRunLoopActivity) { kCFRunLoopEntry = (1UL << 0), // 即將進入Loop kCFRunLoopBeforeTimers = (1UL << 1), // 即將處理 Timer kCFRunLoopBeforeSources = (1UL << 2), // 即將處理 Source kCFRunLoopBeforeWaiting = (1UL << 5), // 即將進入休眠 kCFRunLoopAfterWaiting = (1UL << 6), // 剛從休眠中喚醒 kCFRunLoopExit = (1UL << 7), // 即將退出Loop };

三、Runloop內部邏輯:關鍵在兩個判斷點(是否睡覺,是否退出)

  • 代碼實現:

// RunLoop的實現 int CFRunLoopRunSpecific(runloop, modeName, seconds, stopAfterHandle) { // 0.1 根據modeName找到對應mode CFRunLoopModeRef currentMode = __CFRunLoopFindMode(runloop, modeName, false); // 0.2 若是mode裏沒有source/timer/observer, 直接返回。   if (__CFRunLoopModeIsEmpty(currentMode)) return; // 1.1 通知 Observers: RunLoop 即將進入 loop。---(OB會建立釋放池) __CFRunLoopDoObservers(runloop, currentMode, kCFRunLoopEntry); // 1.2 內部函數,進入loop __CFRunLoopRun(runloop, currentMode, seconds, returnAfterSourceHandled) {   Boolean sourceHandledThisLoop = NO; int retVal = 0; do { // 2.1 通知 Observers: RunLoop 即將觸發 Timer 回調。 __CFRunLoopDoObservers(runloop, currentMode, kCFRunLoopBeforeTimers); // 2.2 通知 Observers: RunLoop 即將觸發 Source0 (非port) 回調。 __CFRunLoopDoObservers(runloop, currentMode, kCFRunLoopBeforeSources); // 執行被加入的block __CFRunLoopDoBlocks(runloop, currentMode);   // 2.3 RunLoop 觸發 Source0 (非port) 回調。 sourceHandledThisLoop = __CFRunLoopDoSources0(runloop, currentMode, stopAfterHandle); // 執行被加入的block __CFRunLoopDoBlocks(runloop, currentMode);   // 2.4 若是有 Source1 (基於port) 處於 ready 狀態,直接處理這個 Source1 而後跳轉去處理消息。 if (__Source0DidDispatchPortLastTime) { Boolean hasMsg = __CFRunLoopServiceMachPort(dispatchPort, &msg) if (hasMsg) goto handle_msg; } // 3.1 若是沒有待處理消息,通知 Observers: RunLoop 的線程即將進入休眠(sleep)。--- (OB會銷燬釋放池並創建新釋放池) if (!sourceHandledThisLoop) { __CFRunLoopDoObservers(runloop, currentMode, kCFRunLoopBeforeWaiting); }  // 3.2. 調用 mach_msg 等待接受 mach_port 的消息。線程將進入休眠, 直到被下面某一個事件喚醒。 // - 一個基於 port 的Source1 的事件。 // - 一個 Timer 到時間了 // - RunLoop 啓動時設置的最大超時時間到了 // - 被手動喚醒 __CFRunLoopServiceMachPort(waitSet, &msg, sizeof(msg_buffer), &livePort) { mach_msg(msg, MACH_RCV_MSG, port); // thread wait for receive msg } // 3.3. 被喚醒,通知 Observers: RunLoop 的線程剛剛被喚醒了。 __CFRunLoopDoObservers(runloop, currentMode, kCFRunLoopAfterWaiting); // 4.0 處理消息。 handle_msg: // 4.1 若是消息是Timer類型,觸發這個Timer的回調。 if (msg_is_timer) { __CFRunLoopDoTimers(runloop, currentMode, mach_absolute_time()) } // 4.2 若是消息是dispatch到main_queue的block,執行block。 else if (msg_is_dispatch) { __CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__(msg); } // 4.3 若是消息是Source1類型,處理這個事件 else { CFRunLoopSourceRef source1 = __CFRunLoopModeFindSourceForMachPort(runloop, currentMode, livePort); sourceHandledThisLoop = __CFRunLoopDoSource1(runloop, currentMode, source1, msg); if (sourceHandledThisLoop) { mach_msg(reply, MACH_SEND_MSG, reply); } } // 執行加入到Loop的block __CFRunLoopDoBlocks(runloop, currentMode); // 5.1 若是處理事件完畢,啓動Runloop時設置參數爲一次性執行,設置while參數退出Runloop if (sourceHandledThisLoop && stopAfterHandle) { retVal = kCFRunLoopRunHandledSource; // 5.2 若是啓動Runloop時設置的最大運轉時間到期,設置while參數退出Runloop } else if (timeout) { retVal = kCFRunLoopRunTimedOut; // 5.3 若是啓動Runloop被外部調用強制中止,設置while參數退出Runloop } else if (__CFRunLoopIsStopped(runloop)) { retVal = kCFRunLoopRunStopped; // 5.4 若是啓動Runloop的modeItems爲空,設置while參數退出Runloop } else if (__CFRunLoopModeIsEmpty(runloop, currentMode)) { retVal = kCFRunLoopRunFinished; } // 5.5 若是沒超時,mode裏沒空,loop也沒被中止,那繼續loop,回到第2步循環。 } while (retVal == 0); } // 6. 若是第6步判斷後loop退出,通知 Observers: RunLoop 退出。--- (OB會銷燬新釋放池) __CFRunLoopDoObservers(rl, currentMode, kCFRunLoopExit); }
  • 函數做用棧顯示:

{ // 1.1 通知Observers,即將進入RunLoop // 此處有Observer會建立AutoreleasePool: _objc_autoreleasePoolPush(); __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__(kCFRunLoopEntry); do { // 2.1 通知 Observers: 即將觸發 Timer 回調。 __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__(kCFRunLoopBeforeTimers); // 2.2 通知 Observers: 即將觸發 Source (非基於port的,Source0) 回調。 __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__(kCFRunLoopBeforeSources); // 執行Block __CFRUNLOOP_IS_CALLING_OUT_TO_A_BLOCK__(block); // 2.3 觸發 Source0 (非基於port的) 回調。 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__(source0); // 執行Block __CFRUNLOOP_IS_CALLING_OUT_TO_A_BLOCK__(block); // 3.1 通知Observers,即將進入休眠 // 此處有Observer釋放並新建AutoreleasePool: _objc_autoreleasePoolPop(); _objc_autoreleasePoolPush(); __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__(kCFRunLoopBeforeWaiting); // 3.2 sleep to wait msg. mach_msg() -> mach_msg_trap(); // 3.3 通知Observers,線程被喚醒 __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__(kCFRunLoopAfterWaiting); // 4.1 若是是被Timer喚醒的,回調Timer __CFRUNLOOP_IS_CALLING_OUT_TO_A_TIMER_CALLBACK_FUNCTION__(timer); // 4.2 若是是被dispatch喚醒的,執行全部調用 dispatch_async 等方法放入main queue 的 block __CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__(dispatched_block); // 4.3 若是若是Runloop是被 Source1 (基於port的) 的事件喚醒了,處理這個事件 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__(source1); // 5. 退出判斷函數調用棧無顯示 } while (...); // 6. 通知Observers,即將退出RunLoop // 此處有Observer釋放AutoreleasePool: _objc_autoreleasePoolPop(); __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__(kCFRunLoopExit); }

一步一步寫具體的實現邏輯過於繁瑣不便理解,按Runloop狀態大體分爲:

1- Entry:通知OB(建立pool); 2- 執行階段:按順序通知OB並執行timer,source0;如有source1執行source1; 3- 休眠階段:利用mach_msg判斷進入休眠,通知OB(pool的銷燬重建);被消息喚醒通知OB; 4- 執行階段:按消息類型處理事件; 5- 判斷退出條件:若是符合退出條件(一次性執行,超時,強制中止,modeItem爲空)則退出,不然回到第2階段; 6- Exit:通知OB(銷燬pool)。

四、Runloop本質:mach port和mach_msg()。

Mach是XNU的內核,進程、線程和虛擬內存等對象經過端口發消息進行通訊,Runloop經過mach_msg()函數發送消息,若是沒有port 消息,內核會將線程置於等待狀態 mach_msg_trap() 。若是有消息,判斷消息類型處理事件,並經過modeItem的callback回調(處理事件的具體執行是在DoBlock裏仍是在回調裏目前我還不太明白???)。

Runloop有兩個關鍵判斷點,一個是經過msg決定Runloop是否等待,一個是經過判斷退出條件來決定Runloop是否循環。


五、如何處理事件:

  • 界面刷新:
    當UI改變( Frame變化、 UIView/CALayer 的繼承結構變化等)時,或手動調用了 UIView/CALayer 的 setNeedsLayout/setNeedsDisplay方法後,這個 UIView/CALayer 就被標記爲待處理。
    蘋果註冊了一個用來監聽BeforeWaiting和Exit的Observer,在它的回調函數裏會遍歷全部待處理的 UIView/CAlayer 以執行實際的繪製和調整,並更新 UI 界面。

  • 事件響應:
    當一個硬件事件(觸摸/鎖屏/搖晃/加速等)發生後,首先由 IOKit.framework 生成一個 IOHIDEvent 事件並由 SpringBoard 接收, 隨後由mach port 轉發給須要的App進程。
    蘋果註冊了一個 Source1 (基於 mach port 的) 來接收系統事件,經過回調函數觸發Sourece0(因此UIEvent其實是基於Source0的),調用 _UIApplicationHandleEventQueue() 進行應用內部的分發。
    _UIApplicationHandleEventQueue() 會把 IOHIDEvent 處理幷包裝成 UIEvent 進行處理或分發,其中包括識別 UIGesture/處理屏幕旋轉/發送給 UIWindow 等。

  • 手勢識別:
    若是上一步的 _UIApplicationHandleEventQueue() 識別到是一個guesture手勢,會調用Cancel方法將當前的touchesBegin/Move/End 系列回調打斷。隨後系統將對應的 UIGestureRecognizer 標記爲待處理。
    蘋果註冊了一個 Observer 監測 BeforeWaiting (Loop即將進入休眠) 事件,其回調函數爲 _UIGestureRecognizerUpdateObserver(),其內部會獲取全部剛被標記爲待處理的 GestureRecognizer,並執行GestureRecognizer的回調。
    當有 UIGestureRecognizer 的變化(建立/銷燬/狀態改變)時,這個回調都會進行相應處理。

  • GCD任務:
    當調用 dispatch_async(dispatch_get_main_queue(), block) 時,libDispatch 會向主線程的 RunLoop 發送消息,RunLoop會被喚醒,並從消息中取得這個 block,並在回調裏執行這個 block。Runloop只處理主線程的block,dispatch 到其餘線程仍然是由 libDispatch 處理的。

  • timer:(見上modeItem部分)

  • 網絡請求:
    關於網絡請求的接口:最底層是CFSocket層,而後是CFNetwork將其封裝,而後是NSURLConnection對CFNetwork進行面向對象的封裝,NSURLSession 是 iOS7 中新增的接口,也用到NSURLConnection的loader線程。因此仍是以NSURLConnection爲例。
    當開始網絡傳輸時,NSURLConnection 建立了兩個新線程:com.apple.NSURLConnectionLoader 和 com.apple.CFSocket.private。其中 CFSocket 線程是處理底層 socket 鏈接的。NSURLConnectionLoader 這個線程內部會使用 RunLoop 來接收底層 socket 的事件,並經過以前添加的 Source0 通知到上層的 Delegate。


六、應用:

  • 滑動與圖片刷新;
    當tableview的cell上有須要從網絡獲取的圖片的時候,滾動tableView,異步線程會去加載圖片,加載完成後主線程就會設置cell的圖片,可是會形成卡頓。可讓設置圖片的任務在CFRunLoopDefaultMode下進行,當滾動tableView的時候,RunLoop是在 UITrackingRunLoopMode 下進行,不去設置圖片,而是當中止的時候,再去設置圖片。

- (void)viewDidLoad { [super viewDidLoad]; // 只在NSDefaultRunLoopMode下執行(刷新圖片) [self.myImageView performSelector:@selector(setImage:) withObject:[UIImage imageNamed:@""] afterDelay:ti inModes:@[NSDefaultRunLoopMode]]; }
  • 常駐子線程,保持子線程一直處理事件
    爲了保證線程長期運轉,能夠在子線程中加入RunLoop,而且給Runloop設置item,防止Runloop自動退出。

    [self.lock unlock];+ (void)networkRequestThreadEntryPoint:(id)__unused object { @autoreleasepool { [[NSThread currentThread] setName:@"AFNetworking"]; NSRunLoop *runLoop = [NSRunLoop currentRunLoop]; [runLoop addPort:[NSMachPort port] forMode:NSDefaultRunLoopMode]; [runLoop run]; } } + (NSThread *)networkRequestThread { static NSThread *_networkRequestThread = nil; static dispatch_once_t oncePredicate; dispatch_once(&oncePredicate, ^{ _networkRequestThread = [[NSThread alloc] initWithTarget:self selector:@selector(networkRequestThreadEntryPoint:) object:nil]; [_networkRequestThread start]; }); return _networkRequestThread; } - (void)start { [self.lock lock]; if ([self isCancelled]) { [self performSelector:@selector(cancelConnection) onThread:[[self class] networkRequestThread] withObject:nil waitUntilDone:NO modes:[self.runLoopModes allObjects]];  } else if ([self isReady]) { self.state = AFOperationExecutingState; [self performSelector:@selector(operationDidStart) onThread:[[self class] networkRequestThread] withObject:nil waitUntilDone:NO modes:[self.runLoopModes allObjects]];  
} } 
相關文章
相關標籤/搜索