導讀:
構造函數(shù)就是類究珊、結構體等在實例化之前的準備過程婴氮。我們在編寫iOS代碼的時候综看,時常要用到它的構造函數(shù)品腹,特別是在自定義一個類的時候。那么如何用好它的構造函數(shù)就成為了關鍵红碑。
我們先來回顧下Objective-C中的構造函數(shù)init:
@interface Person: NSObject
@property (nonatomic, copy) NSString *name;
- (instance)init;
- (instance)initWithName:(NSString *)name;
@end
@implementation Person
- (instancetype)init {
self = [super init];
if (self) {
_name = @"cranz";
}
return self;
}
- (instancetype)initWithName:(NSString *)name {
self = [super init];
if (self) {
_name = name;
}
return self;
}
@end
以上就是簡單OC版初始化方法舞吭,因為所有類都是繼承自NSObject類的,因此要調用父類的初始化方法析珊,以保證所有屬性都被初始化成功[super init]
羡鸥。
再來說說Swift,Swift作為一門融合了眾家之長的新興語言忠寻,必有它的優(yōu)勢所在惧浴。當然這個不是本文要講的東西。我們還是回到它的構造函數(shù)中來奕剃。
Swift的構造過程
首先衷旅,Swift的構造器分為指定構造器和便利構造器捐腿。我們可以聯(lián)想OC中的實例初始化方法和類初始化方法。
我們先看下面簡單的指定構造器:
// 構造函數(shù)又稱為構造器柿顶,在創(chuàng)建某個類或者是結構體的實例的時候被調用茄袖。在Swift中init是個關鍵字,沒有func修飾符嘁锯。最簡單的構造器如下:
// 結構體也是如此
class Person {
var name: String
init() {
name = "Cranz"
}
init(name: String) {
self.name = name
}
}
var p = Person()
p.name
// 結果打印 Cranz
var p1 = Person(name: "Jack")
p1.name
// 結果打印 Jack
Swift中定義的每一個類都是一個新的Swift類宪祥,因此它不需要像OC中那樣每個init方法中都去調用父類的初始化方法。不過如果是一個類繼承自另一個類那就需要重寫了家乘。
class Man: Person {
let gender: String
override init() {
gender = "男性"
// 這里注意蝗羊,我們在不需要給父類的屬性重新復制時,可以省略調用父類的初始化方法仁锯,結果顯示正確耀找,說明底層還是為我們做了下面的操作
// super.init()
}
}
let m = Man()
m.name
m.gender
// 結果打印 Cranz ,男性
然后在上面會出現(xiàn)一個問題业崖,就是假如我們把super.init()寫在gender賦值之前涯呻,如下:
class Man: Person {
let gender: String
override init() {
super.init()
gender = "男性"
}
}
我們可以看到,編譯器提示錯誤腻要,告訴我們gender屬性的初始化必須在super.init()之前。這就奇怪了涝登,我們在OC中都是寫在一開始的雄家,為什么到Swift中就不行了呢。
原因:
看phase1胀滚,蘋果文檔明確告訴我們初始化的過程:
- 類的指定或便利構造器被調用
- 類實例被分配內存趟济,但此時內存還未進行初始化
- 類的指定構造器要保證其引用的存儲屬性有值,這些存儲屬性就是在這時被初始化
- 調用父類的初始化方法
- 上述調用鏈的行為直到頂部
- 一旦到達了調用鏈頂部咽笼,并且最終的類也確保所有存儲屬性初始化完畢顷编。那么此時實例的內存就被認為是初始化完畢,第一階段完成
猜想:
因為Swift要求實例在初始化的時候保證所有存儲屬性被初始化完成剑刑,那么這個檢查應該是在基類中的媳纬。也就是說,有A施掏,B钮惠,C三個類,對應關系是A->B->C七芭,A是基類素挽。我們在初始化C的時候,假如C的存儲屬性寫在super.init()之后狸驳,也就是說在調用鏈達到A的時候预明,編譯器檢測到C的存儲屬性并沒有被初始化完成缩赛,因此,就會報錯撰糠。而將C自身的存儲屬性寫在super.init()之前就可以保證在調用鏈到達頂部的時候確保所有存儲屬性初始化完畢酥馍。
當然這些我沒有進行驗證,只根據(jù)上下文進行了推測窗慎,因此有問題的話或者有直接證明還望告知一二物喷。
其實還有一個便利構造器沒說,下面就提一下遮斥。
class Person {
let name: String
init() {
name = "Cranz"
}
init(name: String) {
self.name = name
}
convenience init(yourName name: String) {
self.init(name: name)
}
}
// 有兩點需要注意
// 1.不要和指定構造器的函數(shù)重合
// 2.必須調用自身的指定構造器
最后貼上一張?zhí)O果的官方圖:
- 意思就是說峦失,指定構造器最終必會調用其父類的指定構造器,直到調用鏈頂部
- 便利構造器必最終要調用到本類的指定構造器完成初始化