self = [super init]

轉(zhuǎn)載:http://blog.csdn.net/aoyuehan11/article/details/10268231

[Obj-C筆記] "self = [super init]"的解釋與潛藏bug
Objective-C的推薦init方法寫法如下:

- (id) init
{
    if(self = [super init])
    {
        //為子類增加屬性進(jìn)行初始化
    }
    return self;
}

這里涉及了幾個(gè)問題,

  1. [super init]的作用:

面向?qū)ο蟮捏w現(xiàn),先利用父類的init方法為子類實(shí)例的父類部分屬性初始化。

  1. self 為什么要賦值為[super init]:

簡單來說是為了防止父類的初始化方法release掉了self指向的空間并重新alloc了一塊空間。這時(shí)的話,[super init]可能alloc失敗,這時(shí)就不再執(zhí)行if中的語句。

  1. super作為消息接受者的實(shí)質(zhì):

super并不是真正的指針,[super message]的實(shí)質(zhì)是由self來接受父類的message。需要注意的是,[super message]中,message方法出現(xiàn)的self為[super message]語境中的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í)編譯能通過,但當(dāng)BObj的實(shí)例使用到BObj擴(kuò)充的屬性時(shí),就會(huì)出現(xiàn)一個(gè)運(yùn)行時(shí)錯(cuò)誤。錯(cuò)誤的原因在于AObj的init方法用[AObj alloc]重新獲得了一塊僅僅適合存放AObj實(shí)例的空間。而BObj的init方法以為這是塊適合存放BObj的空間。當(dāng)試圖讀寫B(tài)Obj的擴(kuò)充屬性時(shí)便會(huì)產(chǎn)生運(yùn)行時(shí)錯(cuò)誤。

因此,當(dāng)init方法需要重新alloc一塊空間時(shí),正確的寫法如下:

- (id) init
{
    id tmp = self;
                    
    self = [[self class] alloc];
                    
    [tmp release];
    //other staffs
    return self;
}

注意第5行,[self class]將獲得self指向的實(shí)例對應(yīng)的類實(shí)例,本例中便是BObj。這樣AObj的任何子類的init方法都能保證安全了。

if( self = [super init] )這是一種通常的建議寫法,賦值并測零只是為了防止超類在初始化過程中發(fā)生改變,返回了不同的對象

為什么一定要 super alloc ?

眾所周知,Objective-C是一門面向?qū)ο蟮恼Z言,一般情況下,我們在Objective-C中定義一個(gè)類時(shí),總要提供一個(gè)初始化方法,一般大家都是這樣寫的:

- (MyClass *)init
{
    self = [super init];
    if (self) {
        //執(zhí)行一些資源、變量的初始化工作
    }
    return self;
}

這樣一段簡單的代碼,卻有很多可以思考的問題:
1、為什么要通過[super init]來調(diào)用父類的初始化方法,父類的初始化方法里又執(zhí)行了什么東西?
首先,我們知道對象繼承的概念,一個(gè)子類從父類繼承,那么也要實(shí)現(xiàn)父類的所有功能,這就是is-a的關(guān)系,比如說狗是哺乳動(dòng)物,那么狗必定具有哺乳動(dòng)物的特征和功能。所以在子類的初始化方法中,必須首先調(diào)用父類的初始化方法,以實(shí)現(xiàn)父類相關(guān)資源的初始化。例如我們在初始化狗這一對象時(shí),必須先初始化哺乳動(dòng)物這一對象,并把結(jié)果賦予狗,以使狗滿足屬于哺乳動(dòng)物這一特征。
典型的,在iOS下,所有的類都繼承于NSObject,而NSObject的init方法很簡單,就是return self。當(dāng)父類的初始化完成之后,即self不為nil的情況下,就可以開始做子類的初始化了。
2、是否一定要提供初始化方法,是否一定要使用init作為初始化方法?
我們在Objective-C中創(chuàng)建一個(gè)對象通常使用

MyClass *newclass = [[MyClass alloc] init];
或者
MyClass *newclass = [Myclass new]; 

new方法是NSObject對象的一個(gè)靜態(tài)方法,根據(jù)apple的文檔,該方法實(shí)際上就是alloc和init方法的組合,實(shí)際上二者是一樣的,但 apple還是推薦我們使用第一種方法,為什么呢?因?yàn)槭褂玫谝环N方法,你可以使用自己定義的init方法來做一些初始化(用自己寫的init*****方法),當(dāng)然,如果子類沒有提供 init方法,自然調(diào)用的就是父類的init方法了。所以說,從安全性的角度來收,作為開發(fā)者我們在對象使用之前是一定要對對象進(jìn)行初始化的,因此在定義類的時(shí)候一定要提供初始化方法。但是否一定要使用init作為方法名呢?答案是不一定。使用init作為方法名只是你重寫了NSObject的init方法而已,如果你自己重新定義一個(gè)初始化方法,也是完全可以的,只要你在使用的時(shí)候記得調(diào)用新定義的初始化方法就可以了。
但是,這種方法從設(shè)計(jì)角度來看我覺得是不可取的。在可復(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ū)ο缶幊讨校绻帉懸粋€(gè)類而沒有包含構(gòu)造函數(shù),這個(gè)類仍能編譯并且完全可以正常使用。如果類沒有提供顯式的構(gòu)造函數(shù),編譯器會(huì)提供一個(gè)默認(rèn)的構(gòu)造函數(shù)給你。除了創(chuàng)建對象本身,默認(rèn)構(gòu)造函數(shù)的唯一工作就是調(diào)用其超類的構(gòu)造函數(shù)。在很多情況下,這個(gè)超類是語言框架的一部分,如Java中的 Object類,objective-c 中的NSObject類。

不論是何種情況,在類中至少包含一個(gè)構(gòu)造函數(shù)是一種很好的編程實(shí)踐,如果類中有屬性,好的實(shí)踐往往是初始化這些屬性。

、、、、、、

另外還有一篇對self = [super init]的貼吧討論粘貼的文章,挺好的,附上鏈接:
http://blog.sina.com.cn/s/blog_6badd5400100wet8.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容