問題描述
對于經(jīng)常使用xib的童鞋們來說秒旋,使用xib來初始化一些簡單的viewController痴柔,簡直是猶如探囊取物。但是铛碑,今天我畫好xib之后狠裹,使用老套路,[[ViewController alloc] init]賦值給視圖控制器汽烦,然后push之后發(fā)現(xiàn)涛菠,我在xib上添加的東西都沒有,這就是遇到的這個問題撇吞。
問題形成原因
經(jīng)過一番的查找俗冻,終于找到了出現(xiàn)問題的原因,原因是我使用的viewController的父類不恰當?shù)闹貙懥薼oadView方法牍颈。
上述問題出現(xiàn)時使用- (instancetype)initWithNibName:(nullable NSString *)nibNameOrNil bundle:(nullable NSBundle *)nibBundleOrNil迄薄;方法可以正常加載自定義的內(nèi)容。那么我們就不得不看一下這個神秘的方法煮岁。
/*
The designated initializer. If you subclass UIViewController, you must call the super implementation of this
method, even if you aren't using a NIB. (As a convenience, the default init method will do this for you,
and specify nil for both of this methods arguments.) In the specified NIB, the File's Owner proxy should
have its class set to your view controller subclass, with the view outlet connected to the main view. If you
invoke this method with a nil nib name, then this class' -loadView method will attempt to load a NIB whose
name is the same as your view controller's class. If no such NIB in fact exists then you must either call
-setView: before -view is invoked, or override the -loadView method to set up your views programatically.
*/
-- (instancetype)initWithNibName:(nullable NSString *)nibNameOrNil bundle:(nullable NSBundle *)nibBundleOrNil NS_DESIGNATED_INITIALIZER;
可以看到讥蔽,蘋果官方的注釋中說到涣易,不管你有沒有使用NIB,這個方法都會被調(diào)用冶伞,此時會將兩個參數(shù)傳為nil新症;并說到,如果nib傳入的參數(shù)為nil响禽,那么loadView方法就會嘗試加載一個NIB徒爹,這個NIB的名字與該viewController的類名相同;如果沒有與之相對應的NIB芋类,那么必須在用view之前設置view或者重寫-loadView方法來自定義view隆嗅。
可能看了這個還是云里霧里,我們來看一下李明杰老師的博客中對loadView方法的介紹:
loadView
1.什么時候被調(diào)用侯繁?
每次訪問UIViewController的view(比如controller.view胖喳、self.view)而且view為nil,loadView方法就會被調(diào)用巫击。
2.有什么作用禀晓?
loadView方法是用來負責創(chuàng)建UIViewController的view
3.默認實現(xiàn)是怎樣的?
默認實現(xiàn)即[super loadView]里面做了什么事情坝锰。
1> 它會先去查找與UIViewController相關(guān)聯(lián)的xib文件,通過加載xib文件來創(chuàng)建UIViewController的view
如果在初始化UIViewController指定了xib文件名重付,就會根據(jù)傳入的xib文件名加載對應的xib文件
[[MJViewController alloc] initWithNibName:@"MJViewController" bundle:nil];
如果沒有明顯地傳xib文件名顷级,就會加載跟UIViewController同名的xib文件
[[MJViewController alloc] init]; // 加載MJViewController.xib
2> 如果沒有找到相關(guān)聯(lián)的xib文件,就會創(chuàng)建一個空白的UIView确垫,然后賦值給UIViewController的view屬性弓颈,大致如下
self.view = [[[UIView alloc] initWithFrame:[UIScreen mainScreen].applicationFrame] autorelease];
// applicationFrame的值是:{{x = 0, y = 20}, {width = 320, height = 460}}
[super loadView]里面就大致完成1>和2>中敘述的內(nèi)容
4.怎樣正確使用這個方法?
大家都知道UIViewController的view可以通過xib文件來創(chuàng)建删掀,但是在某些情況下翔冀,xib不是那么地靈活,所以有時候我們想通過代碼來創(chuàng)建UIView披泪,比如:
self.view = [[[UIWebView alloc] initWithFrame:[UIScreen mainScreen].applicationFrame] autorelease];
如果想通過代碼來創(chuàng)建UIViewController的view纤子,就要重寫loadView方法,并且不需要調(diào)用[super loadView]款票,因為在第3點里面已經(jīng)提到:若沒有xib文件控硼,[super loadView]默認會創(chuàng)建一個空白的UIView。我們既然要通過代碼來自定義UIView艾少,那么就沒必要事先創(chuàng)建一個空白的UIView卡乾,以節(jié)省不必要的開銷。
正確的做法應該是這樣:
- (void)loadView {
self.view = [[[UIWebView alloc] initWithFrame:[UIScreen mainScreen].applicationFrame] autorelease];
}
不需要調(diào)用[super loadView]缚够,你調(diào)用了也不會出錯幔妨,只是造成了一些不必要的開銷鹦赎。
總結(jié)一句話,蘋果設計這個方法就是給我們自定義UIViewController的view用的
正確使用loadView的姿勢
盡量不要在基礎(chǔ)的自定義類中使用loadView误堡,如果要使用钙姊,就一定要保證xib也能正常加載的邏輯。