當我們需要自定義一個View控件時,會有 initWithFrame、initWithCoder、awakeFromNib 這三個系統(tǒng)方法,關(guān)于這三個方法何時調(diào)用,如何調(diào)用,有時候可能很多人會弄混淆。
下面我會就理論結(jié)合代碼舉例來說明一下:
1、initWithFrame
- (instancetype)initWithFrame:(CGRect)frame
{
self = [super initWithFrame:frame];
if (self) {
NSLog(@"initWithFrame");
[self initViews];
}
return self;
}
這個方法是當控件不是從xib、storyboard中創(chuàng)建時,會調(diào)用這個方法,比如:
XibShow *xib = [[XibShow alloc] initWithFrame:CGRectMake(0, 330, KScreenWidth, 100)];
[self.view addSubview:xib];
這里我直接用 [XibShow alloc] initWithFrame 來創(chuàng)建控件,所以,這里自定義控件內(nèi),只會走 initWithFrame 方法,其余兩個不會走。
2、initWithCoder
- (instancetype)initWithCoder:(NSCoder *)coder
{
self = [super initWithCoder:coder];
if (self) {
NSLog(@"initWithCoder");
[self initViews];
}
return self;
}
當控件是從xib、storyboard中創(chuàng)建時,就會調(diào)用這個方法。
概括的理解就是:你的view或cell在xib或storyboard上有體現(xiàn),比如直接用xib創(chuàng)建的,或是拉控件關(guān)聯(lián)已有類等。
而且,即使我們自定義的view是純代碼,沒有xib,但我們在vc的xib里引用了,效果是一樣的。

比如,這兩種方式創(chuàng)建的自定義view,只要我們有在xib或storyboard里引用過,就都會執(zhí)行initWithCoder和awakeFromNib
3、awakeFromNib
-(void)awakeFromNib{
NSLog(@"awakeFromNib");
[super awakeFromNib];
[self initViews];
}
這個方法在initWithCoder:方法后調(diào)用 ,順序是:initWithCoder -> awakeFromNib (下面會有代碼截圖)
- 可以這么來理解:
一開始經(jīng)過initWithCoder創(chuàng)建出來的控件是死的,然后通過awakeFromNib來喚醒,所以這會有一個先后的調(diào)用順序。
下面會有一個具體的示例:
1、創(chuàng)建一個vc,在storyboard上關(guān)聯(lián)界面,在界面上新增一個view,關(guān)聯(lián)我們的自定義view:XibShow。

2、在vc里,再額外用代碼新增一個自定義view控件,add到另一個位置。

這樣,運行出來的結(jié)果如下:


其中:打印順序如上圖所示。
可以發(fā)現(xiàn):
1、在viewDidLoad之前,由于是storyboard上創(chuàng)建的類,會先執(zhí)行好,且會順序走 initWithCoder、awakeFromNib兩個方法
2、到viewDidLoad之后,執(zhí)行到用代碼創(chuàng)建的view,只會走 initWithFrame 方法
PS:這里還有個小細節(jié):我在storyboard上托控件創(chuàng)建的類,即使我給了背景色,但執(zhí)行出來,還是以自定義view本身的背景色為準。
(可以看上圖,開始設(shè)的灰色,但運行出來還是紅色)
最后總結(jié):
1、initWithFrame只適用純代碼創(chuàng)建時調(diào)用,不涉及xib或storyboard。
2、initWithCoder、awakeFromNib是從xib、storyboard中創(chuàng)建時會調(diào)用。
3、所以一般來說,為了安全起見,初始化控件的代碼我們寫在一個方法里面(類似我上面的initViews方法),然后我們分別在initWithFrame和initWithCoder中都調(diào)用一下,這樣就保證了無論你是通過什么方式創(chuàng)建的控件,代碼都能起到作用。