Application 的生命周期

來自 CocoaChina,加一些自己的認識。

1、應用程序的狀態

  • Not running未運行:程序沒啟動。
  • Inactive未激活:程序在前臺運行,不過沒有接收到事件。在沒有事件處理情況下程序通常停留在這個狀態。
  • Active激活:程序在前臺運行而且接收到了事件。這也是前臺的一個正常的模式。
  • Backgroud后臺:程序在后臺而且能執行代碼,大多數程序進入這個狀態后會在在這個狀態上停留一會。時間到之后會進入掛起狀態(Suspended)。有的程序經過特殊的請求后可以長期處于Backgroud狀態。
  • Suspended掛起:程序在后臺不能執行代碼。系統會自動把程序變成這個狀態而且不會發出通知。當掛起時,程序還是停留在內存中的,當系統內存低時,系統就把掛起的程序清除掉,為前臺程序提供更多的內存。

2、各個程序運行狀態時代理的回調

①告訴代理進程啟動但還沒進入狀態保存

- (BOOL)application:(UIApplication *)application willFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    NSLog(@"①告訴代理進程啟動但還沒進入狀態保存"); 
    return YES; 
}

②告訴代理啟動基本完成程序準備開始運行

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { 
    self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]]; 
    NSLog(@"②告訴代理啟動基本完成程序準備開始運行"); 
    // Override point for customization after application launch. 
    self.window.backgroundColor = [UIColor whiteColor]; 
    [self.window makeKeyAndVisible]; 
    return YES; 
} 

其中 launchOptions 是帶有啟動參數的字典,存儲有此程序啟動的原因。launchOptions 中的鍵值對可以看 UIApplication Class Reference 中的 Launch Options Keys 一節。

若用戶直接啟動,launchOptions 內無數據;
若由其他應用程序通過 openURL: 啟動,則 UIApplicationLaunchOptionsURLKey 對應的對象為啟動 URL(NSURL),UIApplicationLaunchOptionsSourceApplicationKey 對應啟動的源應用程序的bundle ID (NSString);
若由本地通知啟動,則 UIApplicationLaunchOptionsLocalNotificationKey 對應的是為啟動應用程序的的本地通知對象(UILocalNotification);
若由遠程通知啟動,則 UIApplicationLaunchOptionsRemoteNotificationKey 對應的是啟動應用程序的的遠程通知信息 userInfo(NSDictionary);
其他key還有 UIApplicationLaunchOptionsAnnotationKey、UIApplicationLaunchOptionsLocationKey、
UIApplicationLaunchOptionsNewsstandDownloadsKey。

如果要在啟動時,做出一些區分,那就需要在下面的代碼做處理。
比如應用可以被某個其它應用調起,那可能要在啟動代碼的地方做出一些驗證,比如以下例子。

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    NSURL *url = [options objectForKey:UIApplicationLaunchOptionsURLKey];
    if (url) {
        // do something.
    }
    NSString *bundleId = [options objectForKey:UIApplicationLaunchOptionsSourceApplicationKey];
    if (bundleId) {
        // do something.
    }
    UILocalNotification * localNotify = [options objectForKey:UIApplicationLaunchOptionsLocalNotificationKey];
    if (localNotify) {
        // do something.
    }
    NSDictionary * userInfo = [options objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey];
    if (userInfo) {
        // do something.
    }
}

③當應用程序將要入非活動狀態執行,在此期間,應用程序不接收消息或事件,比如來電話

- (void)applicationWillResignActive:(UIApplication *)application { 
    // Sent when the application is about to move from active to inactive state. This can occur for certain types of temporary interruptions (such as an incoming phone call or SMS message) or when the user quits the application and it begins the transition to the background state. 
    // Use this method to pause ongoing tasks, disable timers, and throttle down OpenGL ES frame rates. Games should use this method to pause the game. 
    NSLog(@"③當應用程序將要入非活動狀態執行,在此期間,應用程序不接收消息或事件,比如來電話"); 
} 

④當應用程序進入活動狀態執行

- (void)applicationDidBecomeActive:(UIApplication *)application { 
    // Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface. 
    NSLog(@"④當應用程序進入活動狀態執行"); 
} 

⑤當程序被推送到后臺的時候調用。所以要設置后臺繼續運行,則在這個函數里面設置即可

- (void)applicationDidEnterBackground:(UIApplication *)application { 
    // Use this method to release shared resources, save user data, invalidate timers, and store enough application state information to restore your application to its current state in case it is terminated later. 
    // If your application supports background execution, this method is called instead of applicationWillTerminate: when the user quits. 
    NSLog(@"⑤當程序被推送到后臺的時候調用"); 
    [application beginBackgroundTaskWithExpirationHandler:^{ 
        NSLog(@"begin Background Task With Expiration Handler"); 
    }]; 
}

⑥當程序從后臺將要重新回到前臺時候調用

- (void)applicationWillEnterForeground:(UIApplication *)application { 
     // Called as part of the transition from the background to the inactive state; here you can undo many of the changes made on entering the background. 
     NSLog(@"⑥當程序從后臺將要重新回到前臺時候調用"); 
 }

⑦當程序將要退出是被調用,通常是用來保存數據和一些退出前的清理工作。這個需要要設置UIApplicationExitsOnSuspend的鍵值

- (void)applicationWillTerminate:(UIApplication *)application { 
    // Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:. 
    NSLog(@"⑦當程序將要退出是被調用"); 
 }

⑧當程序載入后執行

- (void)applicationDidFinishLaunching:(UIApplication *)application { 
     NSLog(@"⑧當程序載入后執行"); 
 }

程序啟動時:
2014-07-01 15:55:14.706 LifeCycle[5845:60b] ①告訴代理進程啟動但還沒進入狀態保存
2014-07-01 15:55:14.708 LifeCycle[5845:60b] ②告訴代理啟動基本完成程序準備開始運行
2014-07-01 15:55:14.709 LifeCycle[5845:60b] ④當應用程序進入活動狀態執行

按下Home鍵返回主界面:
2014-07-01 15:56:11.756 LifeCycle[5845:60b] ③當應用程序將要入非活動狀態執行
2014-07-01 15:56:11.814 LifeCycle[5845:60b] ⑤當程序被推送到后臺的時候調用

再次打開程序:
2014-07-01 15:57:19.200 LifeCycle[5845:60b] ⑥當程序從后臺將要重新回到前臺時候調用
2014-07-01 15:57:19.201 LifeCycle[5845:60b] ④當應用程序進入活動狀態執行


3、加載應用程序進入前臺


4、加載應用程序進入后臺


5、基于警告式響應中斷


當出現這種中斷時,我們需要在- (void)applicationWillResignActive:(UIApplication *)application方法中進行如下操作:

①停止timer 和其他周期性的任務

②停止任何正在運行的請求

③暫停視頻的播放

④如果是游戲那就暫停它

⑤減少OpenGL ES的幀率

⑥掛起任何分發的隊列和不重要的操作隊列(你可以繼續處理網絡請求或其他時間敏感的后臺任務)

當程序回到active狀態,我們需要在- (void)applicationDidBecomeActive:(UIApplication *)application方法中重新開始上述任務。不過游戲要回到暫停狀態,不能自動開始。


6、進入后臺運行


當應用程序進入后臺時,我們應該做些什么?

保存用戶數據或狀態信息,所有沒寫到磁盤的文件或信息,在進入后臺時,最后都寫到磁盤去,因為程序可能在后臺被殺死。

釋放盡可能釋放的內存。

- (void)applicationDidEnterBackground:(UIApplication *)application

方法有大概5秒的時間讓你完成這些任務。如果超過時間還有未完成的任務,你的程序就會被終止而且從內存中清除。

如果還需要長時間的運行任務,可以在該方法中調用

[application beginBackgroundTaskWithExpirationHandler:^{
    NSLog(@"begin Background Task With Expiration Handler");
}];

應用程序在后臺時的內存使用:請求后臺運行時間和啟動線程來運行長時間運行的任務。

在后臺時,每個應用程序都應該釋放最大的內存。系統努力的保持更多的應用程序在后臺同時 運行。不過當內存不足時,會終止一些掛起的程序來回收內存,那些內存最大的程序首先被終止。

事實上,應用程序應該的對象如果不再使用了,那就應該盡快的去掉強引用,這樣編譯器可以回收這些內存。如果你想緩存一些對象提升程序的性能,你可以在進入后臺時,把這些對象去掉強引用。

下面這樣的對象應該盡快的去掉強引用:

①圖片對象

②你可以重新加載的 大的視頻或數據文件

③任何沒用而且可以輕易創建的對象

在后臺時,為了減少程序占用的內存,系統會自動在回收一些系統幫助你開辟的內存。比如:

①系統回收Core Animation的后備存儲。

②去掉任何系統引用的緩存圖片

③去掉系統管理數據緩存強引用


7、返回前臺運行

在暫停狀態的應用程序必須準備處理任何排隊的通知時,它返回到前臺或后臺執行狀態。暫停的應用程序不執行任何代碼,因此不能處理與方向的變化,時間的變化,偏好的變化,以及許多其他會影響應用程序的外觀或狀態的通知。為了確保這些更改不會丟失,系統排隊許多相關的通知,并把它們傳遞給應用程序,只要它開始再次執行代碼(無論是在前景或背景)。為了防止由偏快轉為超載與它恢復時通知您的應用程序,該系統凝聚事件,并提供一個單一的通知(每個相關類型),反映了凈變化,因為你的應用程序被暫停。


8、程序終止

程序只要符合以下情況之一,只要進入后臺或掛起狀態就會終止:

①iOS4.0以前的系統

②app是基于iOS4.0之前系統開發的。

③設備不支持多任務

④在Info.plist文件中,程序包含了 UIApplicationExitsOnSuspend 鍵。

app如果終止了,系統會調用app的代理的方法 - (void)applicationWillTerminate:(UIApplication *)application,這樣可以讓你可以做一些清理工作。你可以保存一些數據或app的狀態。這個方法也有5秒鐘的限制。超時后方法會返回程序從內存中清除。

注意:用戶可以手工關閉應用程序。


9、The Main Run Loop 主運行循環

Main Run Loop負責處理用戶相關的事件。UIApplication對象在程序啟動時啟動main run Loop,它處理事件和更新視圖的界面。看Main Run Loop就知道,它是運行在程序的主線程上的。這樣保證了接收到用戶相關操作的事件是按順序處理的。

用戶操作設備,相關的操作事件被系統生成并通過UIKit的指定端口分發。事件在內部排成隊列,一個個的分發到Main run loop 去做處理。UIApplication對象是第一個接收到時間的對象,它決定事件如何被處理。觸摸事件分發到主窗口,窗口再分發到對應出發觸摸事件的View。其他的事件通過其他途徑分發給其他對象變量做處理。

大部分的事件可以在你的應用里分發,類似于觸摸事件,遠程操控事件(線控耳機等)都是由app的 responder objects 對象處理的。Responder objects 在你的app里到處都是,比如:UIApplication 對象,view對象,view controller 對象,都是resopnder objects。大部分事件的目標都指定了resopnder object,不過事件也可以傳遞給其他對象。比如,如果view對象不處理事件,可以傳給父類view或者view controller。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,565評論 6 539
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,115評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,577評論 0 382
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,514評論 1 316
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,234評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,621評論 1 326
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,641評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,822評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,380評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,128評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,319評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,879評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,548評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,970評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,229評論 1 291
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,048評論 3 397
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,285評論 2 376

推薦閱讀更多精彩內容