最近有一個小師弟問我生命周期和程序執(zhí)行順序的問題,話不多少,這就分享一篇文章.非常詳細.
當一個視圖控制器被創(chuàng)建,并在屏幕上顯示的時候。 代碼的執(zhí)行順序
1、 alloc ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 創(chuàng)建對象,分配空間
2、init (initWithNibName) 初始化對象,初始化數(shù)據(jù)
3、loadView ? ? ? ? ? ? ? ? ? ? ? ? ?從nib載入視圖 ,通常這一步不需要去干涉。除非你沒有使用xib文件創(chuàng)建視圖
4、viewDidLoad ? ? ? ? ? ? ? ? ? 載入完成,可以進行自定義數(shù)據(jù)以及動態(tài)創(chuàng)建其他控件
5、viewWillAppear ? ? ? ? ? ? ?視圖將出現(xiàn)在屏幕之前,馬上這個視圖就會被展現(xiàn)在屏幕上了
6、viewDidAppear ? ? ? ? ? ? ? 視圖已在屏幕上渲染完成
當一個視圖被移除屏幕并且銷毀的時候的執(zhí)行順序,這個順序差不多和上面的相反
1、viewWillDisappear ? ? ? ? ? ?視圖將被從屏幕上移除之前執(zhí)行
2、viewDidDisappear ? ? ? ? ? ? 視圖已經(jīng)被從屏幕上移除,用戶看不到這個視圖了
3、dealloc ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 視圖被銷毀,此處需要對你在init和viewDidLoad中創(chuàng)建的對象進行釋放
關于viewDidUnload :在發(fā)生內存警告的時候如果本視圖不是當前屏幕上正在顯示的視圖的話, viewDidUnload將會被執(zhí)行,本視圖的所有子視圖將被銷毀,以釋放內存,此時開發(fā)者需要手動對viewLoad、viewDidLoad中創(chuàng)建的對象釋放內存。 因為當這個視圖再次顯示在屏幕上的時候,viewLoad、viewDidLoad?再次被調用,以便再次構造視圖。
當我們創(chuàng)建一個UIViewController類的對象時,通常系統(tǒng)會生成幾個默認的方法,這些方法大多與視圖的調用有關,但是在視圖調用時,這些方法的調用順序如何,需要整理下。
通常上述方法包括如下幾種,這些方法都是UIViewController類的方法:
- (void)viewDidLoad;
- (void)viewDidUnload;
- (void)viewWillAppear:(BOOL)animated;
- (void)viewDidAppear:(BOOL)animated;
- (void)viewWillDisappear:(BOOL)animated;
- (void)viewDidDisappear:(BOOL)animated;
下面介紹下APP在運行時的調用順序。
1)- (void)viewDidLoad;
一個APP在載入時會先通過調用loadView方法或者載入IB中創(chuàng)建的初始界面的方法,將視圖載入到內存中。然后會調用viewDidLoad方法來進行進一步的設置。通常,我們對于各種初始數(shù)據(jù)的載入,初始設定等很多內容,都會在這個方法中實現(xiàn),所以這個方法是一個很常用,很重要的方法。
但是要注意,這個方法只會在APP剛開始加載的時候調用一次,以后都不會再調用它了,所以只能用來做初始設置。
2) - (void)viewDidUnload;
在內存足夠的情況下,軟件的視圖通常會一直保存在內存中,但是如果內存不夠,一些沒有正在顯示的viewcontroller就會收到內存不夠的警告,然后就會釋放自己擁有的視圖,以達到釋放內存的目的。但是系統(tǒng)只會釋放內存,并不會釋放對象的所有權,所以通常我們需要在這里將不需要在內存中保留的對象釋放所有權,也就是將其指針置為nil。
這個方法通常并不會在視圖變換的時候被調用,而只會在系統(tǒng)退出或者收到內存警告的時候才會被調用。但是由于我們需要保證在收到內存警告的時候能夠對其作出反應,所以這個方法通常我們都需要去實現(xiàn)。
另外,即使在設備上按了Home鍵之后,系統(tǒng)也不一定會調用這個方法,因為IOS4之后,系統(tǒng)允許將APP在后臺掛起,并將其繼續(xù)滯留在內存中,因此,viewcontroller并不會調用這個方法來清除內存。
3)- (void)viewWillAppear:(BOOL)animated;
系統(tǒng)在載入所有數(shù)據(jù)后,將會在屏幕上顯示視圖,這時會先調用這個方法。通常我們會利用這個方法,對即將顯示的視圖做進一步的設置。例如,我們可以利用這個方法來設置設備不同方向時該如何顯示。
另外一方面,當APP有多個視圖時,在視圖間切換時,并不會再次載入viewDidLoad方法,所以如果在調入視圖時,需要對數(shù)據(jù)做更新,就只能在這個方法內實現(xiàn)了。所以這個方法也非常常用。
4) - (void)viewDidAppear:(BOOL)animated;
有時候,由于一些特殊的原因,我們不能在viewWillApper方法里,對視圖進行更新。那么可以重寫這個方法,在這里對正在顯示的視圖進行進一步的設置。
5) - (void)viewWillDisappear:(BOOL)animated;
在視圖變換時,當前視圖在即將被移除、或者被覆蓋時,會調用這個方法進行一些善后的處理和設置。
由于在IOS4之后,系統(tǒng)允許將APP在后臺掛起,所以在按了Home鍵之后,系統(tǒng)并不會調用這個方法,因為就這個APP本身而言,APP顯示的view,仍是掛起時候的view,所以并不會調用這個方法。
6) - (void)viewDidDisappear:(BOOL)animated;
我們可以重寫這個方法,對已經(jīng)消失,或者被覆蓋,或者已經(jīng)隱藏了的視圖做一些其他操作。
上述方法的流程圖可以簡單用如下表示:
運行APP —> 載入視圖 —> 調用viewDidLoad方法 —> 調用viewWillAppear方法 —> 調用viewDidAppear方法?—>?? 正常運行
aaaaaaaaA????????????????????????????????????????????????????????????????????????????|
aaaaaaaa|????????????????????????????????????????????? ???????????? ??????? ??????? ? |
aaaaaaaa|?載入新的View?????????????????????????????????????????????? ?????????? |
aaaaaaaa|??????????????????????????????????????????????????????????????? ???????????? |
aaaaaaaa|????????????????????????????????????????????????????? ?????????? ?????? ??? ?V
釋放對象所有權?<—?調用viewDidUnload?<— 收到內存警告?<—?調用viewDidDisappear?<—?調用viewWillDisappear?<—? APP需要調用另一個view
IOS程序啟動執(zhí)行順序
http://www.yifeiyang.net/iphone-developer-advanced-3-iphone-application-startup-process/
IOS?開發(fā)?loadView?和?viewDidLoad?的區(qū)別
iPhone開發(fā)必不可少的要用到這兩個方法。 他們都可以用來在視圖載入的時候,初始化一些內容。 但是他們有什么區(qū)別呢?
viewDidLoad 此方法只有當view從nib文件初始化的時候才被調用。
loadView 此方法在控制器的view為nil的時候被調用。 此方法用于以編程的方式創(chuàng)建view的時候用到。 如:
-(void)loadView{
UIView *view =[[UIView alloc]initWithFrame:[UIScreen
mainScreen].applicationFrame];
[view setBackgroundColor:_color];
self.view = view;
[view release];
}
你在控制器中實現(xiàn)了loadView方法,那么你可能會在應用運行的某個時候被內存管理控制調用。 如果設備內存不足的時候, view 控制器會收到didReceiveMemoryWarning的消息。 默認的實現(xiàn)是檢查當前控制器的view是否在使用。如果它的view不在當前正在使用的view hierarchy里面,且你的控制器實現(xiàn)了loadView方法,那么這個view將被release, loadView方法將被再次調用來創(chuàng)建一個新的view。
--------------------------------------------------------------------------------------------------------------------------------------------
Don't read self.view in -loadView.Onlysetit, don'tgetit.
The self.view property accessorcalls-loadView if the view isn't currently loaded. There's your infinite recursion.
The usual way to build the view programmatically in -loadView, as demonstrated in Apple's pre-Interface-Builder examples, is more like this:
UIView*view=[[UIViewalloc]init...];...[view addSubview:whatever];[view addSubview:whatever2];...self.view=view;[view release];
And I don't blame you for not using IB. I've stuck with this method for all of Instapaper and find myself much more comfortable with it than dealing with IB's complexities, interface quirks, and unexpected behind-the-scenes behavior.