構造方法
1.什么是構造方法?
初始化對象的方法.
默認情況下,在 OC 當中創建1個對象分為兩部分(new 做的事):
+alloc:分配內存空間
-init :初始化對象
2.構造方法的作用是?
用作初始化對象的成員變量.
把 C 語言指針初始化為 NULL
把 OC 對象初始化為 nil
把基本數據類型初始化為0
3.構造方法的書寫格式是?
- (instancetype)init
{
self = [super init];
if (self) {
<#statements#>
}
return self;
}
構造方法的結構很重要.
第一步,調用指定的父類初始化函數.該初始化函數返回父類對象的一個經過初始化的實例.
第二步,把返回回來的實例必須將其賦值給特殊的變量 self.
第三步,如果在初始化過程中出現任何問題,初始化函數的協議指定返回一個 nil對象,而不是一個有效的初始化對象.因此,在將父類的初始化函數返回值賦值 self 后,必須檢查 self 是否是 nil. 如果是,就不想要初始化自身的變量了,返回 nil 即可.上面的事例中,我們在 if 語句中將變量賦值給 self 并檢查它是否是 nil.
第四步:構造方法的真正目的除了創建 self 外,還用于初始化對象中的所有數據成員.因此在驗證 self 不是 nil 后,就可以初始化變量了.在初始化變量后,就可以從初始化方法中返回 self.
疑問:在學習和使用構造方法時,我一直心里有個疑問:為什么 self = [super init];蘋果官網對此沒有明確的說明,所有的一切都是開發者的猜測.
下面是我對此經過多次測試,所得到結果的猜測:
1.第一點,[super init],之所以調用父類的 init 方法,是因為如果子類重寫了定義在父類當中的方法,在子類執行過程當中,就不會執行父類當中的該方法.而面向對象編程的最主要特點之一就是:誰的事情,誰去做.子類當中,繼承了父類的所有可繼承的成員變量,那么該些成員變量就應該是父類自己去初始化,而不應該子類替代父類去做初始化操作.所以此處需要調用父類的構造方法.
2.第二點,也就是難點,為什么要將父類初始化后的實例對象賦值給子類當中的 self ???這才是我疑惑非常久的問題.
猜測1.調用父類的初始化方法,肯定要返回一個對象,那么這個對象用誰去接收?應該看 super 當前代表的是誰,是什么類型來確定.
猜測2.如果父類初始化失敗,在子類當中沒有值接收,就沒有辦法進行把控,所以要賦值給一個當前對象.
// 作用:初始化成員變量
-(instancetype)init
{
// 1.調用父類的初始化方法,進行初始化子類繼承自父類的成員變量.
self = [super init];
// 問題:這里要返回一個初始化好的實例對象,那么用什么來接受這個對象???
// 1.1.用父類?
// 1.2.用子類本身?
// 結論:NSLog(@"super-%@",super.class);super 是本類對象,所以用 self 接受
// 3.因為調用父類初始化方法,可能初始化成功,可能初始化失敗,那么如果初始化失敗,就不應該繼續下面的操作
if (self) {
// 3.1.如果父類初始化成功,下面開始初始化子類成員
_name = @"碼奴!";
}
// 4.返回已經初始化好的對象
return self;
}