轉(zhuǎn)載:http://blog.csdn.net/aoyuehan11/article/details/10268231
[Obj-C筆記](méi) "self = [super init]"的解釋與潛藏bug
Objective-C的推薦init方法寫(xiě)法如下:
- (id) init
{
if(self = [super init])
{
//為子類增加屬性進(jìn)行初始化
}
return self;
}
這里涉及了幾個(gè)問(wèn)題,
- [super init]的作用:
面向?qū)ο蟮捏w現(xiàn),先利用父類的init方法為子類實(shí)例的父類部分屬性初始化。
- self 為什么要賦值為[super init]:
簡(jiǎn)單來(lái)說(shuō)是為了防止父類的初始化方法release掉了self指向的空間并重新alloc了一塊空間。這時(shí)的話,[super init]可能alloc失敗,這時(shí)就不再執(zhí)行if中的語(yǔ)句。
- super作為消息接受者的實(shí)質(zhì):
super并不是真正的指針,[super message]的實(shí)質(zhì)是由self來(lái)接受父類的message。需要注意的是,[super message]中,message方法出現(xiàn)的self為[super message]語(yǔ)境中的self,即子類實(shí)例。
潛藏的bug:
假設(shè)有父類AObj與子類BObj。
當(dāng)AObj的init方法如下:
- (id) init
{
id tmp = self;
self = [AObj alloc];
[tmp release];
//other staffs
return self;
}
BObj的init方法如下:
- (id) init
{
if(self = [super init])
{
//other staffs
}
return self;
}
這時(shí)編譯能通過(guò),但當(dāng)BObj的實(shí)例使用到BObj擴(kuò)充的屬性時(shí),就會(huì)出現(xiàn)一個(gè)運(yùn)行時(shí)錯(cuò)誤。錯(cuò)誤的原因在于AObj的init方法用[AObj alloc]重新獲得了一塊僅僅適合存放AObj實(shí)例的空間。而B(niǎo)Obj的init方法以為這是塊適合存放BObj的空間。當(dāng)試圖讀寫(xiě)B(tài)Obj的擴(kuò)充屬性時(shí)便會(huì)產(chǎn)生運(yùn)行時(shí)錯(cuò)誤。
因此,當(dāng)init方法需要重新alloc一塊空間時(shí),正確的寫(xiě)法如下:
- (id) init
{
id tmp = self;
self = [[self class] alloc];
[tmp release];
//other staffs
return self;
}
注意第5行,[self class]將獲得self指向的實(shí)例對(duì)應(yīng)的類實(shí)例,本例中便是BObj。這樣AObj的任何子類的init方法都能保證安全了。
if( self = [super init] )這是一種通常的建議寫(xiě)法,賦值并測(cè)零只是為了防止超類在初始化過(guò)程中發(fā)生改變,返回了不同的對(duì)象
為什么一定要 super alloc ?
眾所周知,Objective-C是一門(mén)面向?qū)ο蟮恼Z(yǔ)言,一般情況下,我們?cè)贠bjective-C中定義一個(gè)類時(shí),總要提供一個(gè)初始化方法,一般大家都是這樣寫(xiě)的:
- (MyClass *)init
{
self = [super init];
if (self) {
//執(zhí)行一些資源、變量的初始化工作
}
return self;
}
這樣一段簡(jiǎn)單的代碼,卻有很多可以思考的問(wèn)題:
1、為什么要通過(guò)[super init]來(lái)調(diào)用父類的初始化方法,父類的初始化方法里又執(zhí)行了什么東西?
首先,我們知道對(duì)象繼承的概念,一個(gè)子類從父類繼承,那么也要實(shí)現(xiàn)父類的所有功能,這就是is-a的關(guān)系,比如說(shuō)狗是哺乳動(dòng)物,那么狗必定具有哺乳動(dòng)物的特征和功能。所以在子類的初始化方法中,必須首先調(diào)用父類的初始化方法,以實(shí)現(xiàn)父類相關(guān)資源的初始化。例如我們?cè)诔跏蓟愤@一對(duì)象時(shí),必須先初始化哺乳動(dòng)物這一對(duì)象,并把結(jié)果賦予狗,以使狗滿足屬于哺乳動(dòng)物這一特征。
典型的,在iOS下,所有的類都繼承于NSObject,而NSObject的init方法很簡(jiǎn)單,就是return self。當(dāng)父類的初始化完成之后,即self不為nil的情況下,就可以開(kāi)始做子類的初始化了。
2、是否一定要提供初始化方法,是否一定要使用init作為初始化方法?
我們?cè)贠bjective-C中創(chuàng)建一個(gè)對(duì)象通常使用
MyClass *newclass = [[MyClass alloc] init];
或者
MyClass *newclass = [Myclass new];
new方法是NSObject對(duì)象的一個(gè)靜態(tài)方法,根據(jù)apple的文檔,該方法實(shí)際上就是alloc和init方法的組合,實(shí)際上二者是一樣的,但 apple還是推薦我們使用第一種方法,為什么呢?因?yàn)槭褂玫谝环N方法,你可以使用自己定義的init方法來(lái)做一些初始化(用自己寫(xiě)的init*****方法),當(dāng)然,如果子類沒(méi)有提供 init方法,自然調(diào)用的就是父類的init方法了。所以說(shuō),從安全性的角度來(lái)收,作為開(kāi)發(fā)者我們?cè)趯?duì)象使用之前是一定要對(duì)對(duì)象進(jìn)行初始化的,因此在定義類的時(shí)候一定要提供初始化方法。但是否一定要使用init作為方法名呢?答案是不一定。使用init作為方法名只是你重寫(xiě)了NSObject的init方法而已,如果你自己重新定義一個(gè)初始化方法,也是完全可以的,只要你在使用的時(shí)候記得調(diào)用新定義的初始化方法就可以了。
但是,這種方法從設(shè)計(jì)角度來(lái)看我覺(jué)得是不可取的。在可復(fù)用性方面會(huì)比較差,如果確有必要定義一些接受不同參數(shù)的初始化方法,我的建議是,先定義一個(gè)init的公用方法,再到其他方法中調(diào)用它,如:
- (id)init // init的公用方法
{
self = [super init];
if (self) {
}
return self;
}
- (id)initWithString:(NSString *)aString
{
[self init];
self.name = aString;
}
- (id)initWithImage:(UIImage *)aImage
{
[self init];
self.image = aImage;
}
補(bǔ)充:
在面向?qū)ο缶幊讨校绻帉?xiě)一個(gè)類而沒(méi)有包含構(gòu)造函數(shù),這個(gè)類仍能編譯并且完全可以正常使用。如果類沒(méi)有提供顯式的構(gòu)造函數(shù),編譯器會(huì)提供一個(gè)默認(rèn)的構(gòu)造函數(shù)給你。除了創(chuàng)建對(duì)象本身,默認(rèn)構(gòu)造函數(shù)的唯一工作就是調(diào)用其超類的構(gòu)造函數(shù)。在很多情況下,這個(gè)超類是語(yǔ)言框架的一部分,如Java中的 Object類,objective-c 中的NSObject類。
不論是何種情況,在類中至少包含一個(gè)構(gòu)造函數(shù)是一種很好的編程實(shí)踐,如果類中有屬性,好的實(shí)踐往往是初始化這些屬性。
、、、、、、
另外還有一篇對(duì)self = [super init]的貼吧討論粘貼的文章,挺好的,附上鏈接:
http://blog.sina.com.cn/s/blog_6badd5400100wet8.html