本文參考的蘋果文檔
Managing Your App's Life Cycle
1. UIApplication
每一個(gè) iOS app 都會有一個(gè)UIApplication的實(shí)例(或者是UIApplication的子類日月,不過這種情況很少很少)慢显。當(dāng)一個(gè)app啟動(dòng)的時(shí)候谁帕,系統(tǒng)會調(diào)用 UIApplicationMain 函數(shù)类腮,這個(gè)函數(shù)會創(chuàng)建一個(gè)單例 UIApplication 對象煌珊,你可以通過調(diào)用 sharedApplication 這個(gè)類方法來訪問這個(gè)單例 UIApplication 對象象泵。
UIApplication 對象的一個(gè)重要作用就是作為處理到來的用戶事件的初始路由(也就是一個(gè)轉(zhuǎn)發(fā)事件的作用):
- 它可以將 UIControl 類對象發(fā)送給它的動(dòng)作消息再派發(fā)給合適的目標(biāo)對象茁影;
- UIApplication 對象維持一個(gè)或多個(gè)UIWindow對象鲫趁,并且可以通過UIWindow對象檢索UIView對象炕桨。
UIApplication類定義了一個(gè)代理饭尝,即UIApplicationDelegate(使用它必須遵守UIApplicationDelegate協(xié)議并且實(shí)現(xiàn)一些方法)。
UIApplication對象會通知UIApplicationDelegate重要的運(yùn)行時(shí)事件(runtime events)献宫,并讓UIApplicationDelegate進(jìn)行反應(yīng)處理钥平,比如,app啟動(dòng)姊途、內(nèi)存過低涉瘾、app終止等知态。
2. UIApplicationMain函數(shù)
蘋果文檔--UIApplicationMain函數(shù)
app的入口為main.m里的main函數(shù)
int main(int argc, char * argv[]) {
@autoreleasepool {
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}
main函數(shù)執(zhí)行并返回的是一個(gè) UIApplicationMain 函數(shù)
int UIApplicationMain(int argc, char * _Nullable *argv, NSString *principalClassName, NSString *delegateClassName);
盡管聲明了返回類型,但此函數(shù)永遠(yuǎn)不會返回立叛。
UIApplicationMain 函數(shù)可以
實(shí)例化 Application 對象
實(shí)例化并設(shè)置 Appledelegate
設(shè)置主線程對應(yīng)的 RunLoop负敏,并且開始處理事件
如果Info.plist文件中指定了要加載的main nib文件,通過 NSMainNibFile key 和一個(gè)有效nib文件名秘蛇,這個(gè)函數(shù)將加載這個(gè)nib文件
3. UIApplicationDelegate
UIApplicationDelegate要和UIApplication一起使用其做。
UIApplicationDelegate里的方法可以讓你響應(yīng)app一些重要的變化,例如赁还,你可以使用UIApplicationDelegate里的一些方法去應(yīng)對app狀態(tài)的改變(比如當(dāng)從前臺移動(dòng)到后臺的時(shí)候)妖泄,也可以對到來的通知進(jìn)行響應(yīng)****。大多數(shù)的情況下秽浇,UIApplicationDelegate是接收這些通知的唯一途徑浮庐。
Xcode默認(rèn)為一個(gè)新項(xiàng)目提供了一個(gè)UIApplicationDelegate(即AppDelegate類),所以你不需要去自定義柬焕。當(dāng)app啟動(dòng)的時(shí)候审残,UIKit會自動(dòng)創(chuàng)建一個(gè)UIApplicationDelegate類的實(shí)例,并且執(zhí)行里面的自定義代碼斑举。
UIApplicationDelegate實(shí)際上是app的根對象(the root object)搅轿,并且和UIApplication對象一樣,都是一個(gè)單例富玷,并且始終存在于運(yùn)行時(shí)(Runtime)璧坟。
UIApplication對象完成app的大部分底層工作,你可以通過UIApplicationDelegate的方法去管理app的整體行為赎懦。
UIApplicationDelegate扮演的關(guān)鍵角色:
- 包含了app的啟動(dòng)代碼雀鹃;
- 響應(yīng)app的狀態(tài)轉(zhuǎn)變;
- 響應(yīng)來自app外部的通知励两,比如內(nèi)存過低警告黎茎、下載完成的通知等等;
- 它可以進(jìn)行狀態(tài)的保存和恢復(fù)当悔;這點(diǎn)可以看Preserving Your App's UI Across Launches
- 它可以響應(yīng)針對app本身的事件傅瞻,并不特定于view或viewController;
- 可以使用它保存app的中央數(shù)據(jù)對象或任何沒有它自己的viewcontroller的一些內(nèi)容盲憎。
3.1 啟動(dòng)應(yīng)用程序
Launch time(啟動(dòng)時(shí)間)是應(yīng)用生命周期重要的一個(gè)點(diǎn)嗅骄。在啟動(dòng)時(shí)間,Appdelegate會負(fù)責(zé)執(zhí)行初始化app的自定義代碼饼疙,例如:Appdelegate會創(chuàng)建app的初始化數(shù)據(jù)結(jié)構(gòu)溺森,注冊所需要的服務(wù),根據(jù)數(shù)據(jù)調(diào)整app的初始化界面。
Appdelegate在Launch time(啟動(dòng)時(shí)間)的一些任務(wù):
- application:willFinishLaunchingWithOptions: and application:didFinishLaunchingWithOptions:儿惫,這兩個(gè)方法里的options參數(shù)是一個(gè)字典澡罚,里面有app被啟動(dòng)的原因,系統(tǒng)會查看啟動(dòng)原因肾请。
- 確定是否繼續(xù)進(jìn)行狀態(tài)恢復(fù)。application:shouldRestoreApplicationState:
- 注冊遠(yuǎn)程通知更胖。 registerForRemoteNotificationTypes:
- 打開發(fā)送到app的網(wǎng)址铛铁。application:openURL:options:
- 為app提供一個(gè)根窗口對象(the root window object)。實(shí)際上却妨,Xcode默認(rèn)提供了一個(gè)UIWindow饵逐,當(dāng)然你也可以自定義一個(gè)keyWindow。
3.2 管理APP的狀態(tài)
AppDelegate的主要工作之一就是響應(yīng)由系統(tǒng)通知的狀態(tài)轉(zhuǎn)換彪标。當(dāng)狀態(tài)轉(zhuǎn)換發(fā)生時(shí)倍权,系統(tǒng)會調(diào)用AppDelegate的相應(yīng)方法。每一種狀態(tài)都有相應(yīng)的規(guī)則去管理app的運(yùn)行行為捞烟,并且AppDelegate會調(diào)用相應(yīng)的方法薄声。
3.2.1 app有五種狀態(tài)
- Not running(未運(yùn)行狀態(tài)):app未啟動(dòng)或者被終止(無論是被系統(tǒng)還是用戶)。
- Inactive(不活躍狀態(tài)):app在前臺運(yùn)行但未接收事件题画。app只在轉(zhuǎn)換到不同狀態(tài)時(shí)會短暫地保持此狀態(tài)默辨。進(jìn)入此狀態(tài)后,app會很快進(jìn)入后臺(Background)或活動(dòng)(Active)狀態(tài)苍息。(打電話時(shí)或者下拉通知欄時(shí)app會進(jìn)入此狀態(tài))
- Active(活動(dòng)狀態(tài)):app在前臺運(yùn)行并且正在接收事件缩幸。處于前臺的app通常狀態(tài)就是Active。
- Background(后臺狀態(tài)):app在屏幕上不可見但是正在執(zhí)行代碼竞思,這是后臺狀態(tài)表谊。當(dāng)用戶退出應(yīng)用后(應(yīng)該是按home鍵),系統(tǒng)會將app在掛起(suspend)前短暫地移動(dòng)到后臺(Background)狀態(tài)盖喷。
- Suspended(掛起狀態(tài)):應(yīng)用程序在內(nèi)存中爆办,但不執(zhí)行代碼。系統(tǒng)會掛起在后臺(Background)狀態(tài)的應(yīng)用程序传蹈。系統(tǒng)可能會為了騰出內(nèi)存空間押逼,將app清除出內(nèi)存。
3.2.2 狀態(tài)轉(zhuǎn)換圖
圖解:
- 應(yīng)用程序(app)在前臺有 Inactive 和 Active 兩種狀態(tài)惦界;
- Not running 在進(jìn)入 Active 前挑格,會短暫停留在 Inactive 狀態(tài);
- Active 狀態(tài)進(jìn)入 Background 狀態(tài)前沾歪,會短暫地保持 Inactive 狀態(tài)漂彤;
- Background 狀態(tài)會迅速進(jìn)入 Suspended 狀態(tài);
- Suspended 狀態(tài)下系統(tǒng)有可能將其清除,進(jìn)入 Not running 狀態(tài)挫望。
3.3 狀態(tài)轉(zhuǎn)換時(shí)立润,系統(tǒng)會調(diào)用的方法
在狀態(tài)轉(zhuǎn)換時(shí)會調(diào)用下列方法,以下方法都是Appdelegate的方法:
- 啟動(dòng)應(yīng)用程序時(shí)(Launch time)媳板,
application:willFinishLaunchingWithOptions:
application:didFinishLaunchingWithOptions:
- 轉(zhuǎn)換到Active狀態(tài)桑腮,
applicationDidBecomeActive:
- 轉(zhuǎn)換到Background狀態(tài),
applicationDidEnterBackground:
- 轉(zhuǎn)換到Inactive狀態(tài)
applicationWillResignActive:(當(dāng)離開前臺狀態(tài)的時(shí)候調(diào)用)
applicationWillEnterForeground:(當(dāng)從后臺狀態(tài)轉(zhuǎn)到前臺狀態(tài)時(shí)調(diào)用)
- 終止
applicationWillTerminate:(這個(gè)方法只會在app運(yùn)行時(shí)調(diào)用蛉幸,app被掛起時(shí)不會被調(diào)用)
3.4
初次啟動(dòng)app會調(diào)用:
application:willFinishLaunchingWithOptions:
application:didFinishLaunchingWithOptions:
applicationDidBecomeActive:
按下home鍵(從前臺退出到Background狀態(tài)):
applicationWillResignActive:(當(dāng)離開前臺狀態(tài)的時(shí)候調(diào)用)
applicationDidEnterBackground:
再次點(diǎn)擊進(jìn)入app:
applicationWillEnterForeground:(當(dāng)從后臺狀態(tài)轉(zhuǎn)到前臺狀態(tài)時(shí)調(diào)用)
applicationWillResignActive:(當(dāng)離開前臺狀態(tài)的時(shí)候調(diào)用)
4. 生命周期
圖解:
點(diǎn)擊app按鈕破讨,應(yīng)用程序啟動(dòng);
執(zhí)行main函數(shù)奕纫,在main函數(shù)里提陶,執(zhí)行UIApplicationMain函數(shù);
UIApplicationMain函數(shù)會初始化Application對象匹层,并設(shè)置一個(gè)ApplicationDelegate隙笆;
在ApplicationDelegate里,會通過一些方法去處理系統(tǒng)的一些事件升筏;
UIApplication對象負(fù)責(zé)接收系統(tǒng)的事件撑柔,直到應(yīng)用程序退出。
注意:UIApplication負(fù)責(zé)接收系統(tǒng)事件并處理大部分的底層工作仰冠,UIApplicationDelegate可以響應(yīng)一些系統(tǒng)的重要變化(比如狀態(tài)的改變乏冀、通知的到來等等)。