ViewController的生命週期分析和使用

iOS的SDK中提供很多原生ViewController,大大提高了我們的開發效率,下面是我的一些經驗。

一、結構

按結構可以對iOS的所有ViewController分成兩類:
1、主要用於展示內容的ViewController,這種ViewController主要用於爲用戶展示內容,並與用戶交互,如UITableViewController,UIViewController。
2、用於控制和顯示其他ViewController的ViewController。這種ViewController一般都是一個ViewController的容器。如UINavigationController,UITabbarController。它們都有一個屬性:viewControllers。其中UINavigationController表示一種Stack式結構,push一個ViewController或pop一次,因此後一個ViewController一般會依賴前一個ViewController。而UITabbarController表示一個Array結構,各個ViewController是並列的。
第一種ViewController會經常被繼承,用來顯示不同的數據給用戶。而第二種很少被繼承,除非你真的需要自定義它。
注:細心的同學應該能發現,在Xcode中新建一個ViewController時,只可以選擇繼承自UIViewController和UITableViewController,而它們都是第一種。

圖1

二、Controller和View的生命週期

這裏指的View是指Controller的View。它作爲Controler的屬性,生命週期在Controller的生命週期內。就是說你的Controller不能在view釋放前就釋放了。

圖2 ViewController生命週期



當你alloc並init了一個ViewController時,這個ViewController應該是還沒有創建view的。ViewController的view是使用了lazyInit方式創建,就是說你調用的view屬性的getter:[self view]。在getter裏會先判斷view是否創建,如果沒有創建,那麼會調用loadView來創建view。loadView完成時會繼續調用viewDidLoad。loadView和viewDidLoad的一個區別就是:loadView時還沒有view。而viewDidLoad時view以及創建好了。
當view被添加其他view中之前時,會調用viewWillAppear,而之後會調用viewDidAppear。
當view從其他view中移出之前時,會調用viewWillDisAppear,而之後會調用viewDidDisappear。
當view不在使用,而且是disappeared,受到內存警告時,那麼viewController會將view釋放並將其指向nil。

三、代碼組織(如何設計良好的viewcontroller)

ViewController生命週期中有那麼多函數,一個重要問題就是什麼代碼該寫在什麼地方。
1、init裏不要出現創建view的代碼。良好的設計,在init裏應該只有相關數據的初始化,而且這些數據都是比較關鍵的數據。init裏不要掉self.view,否則會導致viewcontroller創建view。(因爲view是lazyinit的)。
2、loadView中只初始化view,一般用於創建比較關鍵的view如tableViewController的tabView,UINavigationController的navgationBar,不可掉用view的getter(在掉super loadView前),最好也不要初始化一些非關鍵的view。如果你是從nib文件中創建的viewController在這裏一定要首先調用super的loadView方法,但建議不要重載這個方法。
3、viewDidLoad 這時候view已經有了,最適合創建一些附加的view和控件了。有一點需要注意的是,viewDidLoad會調用多次(viewcontroller可能多次載入view,參見圖2)。
4、viewWillAppear 這個一般在view被添加到superview之前,切換動畫之前調用。在這裏可以進行一些顯示前的處理。比如鍵盤彈出,一些特殊的過程動畫(比如狀態條和navigationbar顏色)。
5、viewDidAppear 一般用於顯示後,在切換動畫後,如果有需要的操作,可以在這裏加入相關代碼。
6、viewDidUnload 這時候viewController的view已經是nil了。由於這一般發生在內存警告時,所以在這裏你應該將那些不在顯示的view釋放了。比如你在viewcontroller的view上加了一個label,而且這個label是viewcontroller的屬性,那麼你要把這個屬性設置成nil,以免佔用不必要的內存,而這個label在viewDidLoad時會重新創建。

發佈了4 篇原創文章 · 獲贊 3 · 訪問量 1萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章