OC中self與super

個人理解

@1.   self是類的隱藏的參數,指向當前調用方法的對象(類也是對象,類對象),另一個隱藏參數是_cmd,代表當前類方法的selector。

@2.   super並不是隱藏的參數,它只是一個"編譯器指示符",super 就是個障眼法,總而言之》 self與super指向的都是相同的消息接受者

@3.   當使用self調用方法時,會從當前類的方法列表中開始尋找,如果沒有,就從父類中再找,而使用super時,則從父類的方法列表中開始找,然後調用父類的這個方法。注: 從super出現的在的方法所在的類的父類開始查找。

例如在子類裏實現如下方法:

  1. NSLog(@"class is %@", [self class]);  
  2. NSLog(@"class is %@", [super class]);
輸出結果都是子類名。
當 發送 class 消息 時不管是 self  還是 super 其消息主體依然是  self ,也就是說 self 和 super 指向的 是同一個對象。只是 查找方法的位置 區別,一個從本類,一個從本類的超類。

一般情況下  class 方法 只有在 根類 NSObject 中定義,極少情況有子類重寫 class 方法,
所以 [self class] 和 [super class] 都是在 根類中 找方法實現, 消息接收主體都是調用對象。
如果重寫可能會不一樣。

參考博文:

@implementation Son : Father
- (id)init
{
    self = [super init];
    if (self)
    {
       
    }
    return self;
}
複製代碼

這個其實就是 在子類實現初始化前 調用父類的init實現.


首先,要了解 

    1.  self  是什麼?  super 是什麼?

    2.  [ super init] 都做了什麼?

    3.  爲什麼要 self =  [super init];


@1.  self  是什麼 ,super 是什麼 ?

   > 在動態方法(對象方法)中,self代表着"對象"

   > 在靜態方法(類方法)中,self代表着"類"

   > 萬變不離其宗,記住一句話就行了:self代表着當前方法的調用者

self 和 super 是oc 提供的 兩個保留字。 但有根本區別,

    self是類的隱藏的參數變量,指向當前調用方法的對象(類也是對象,類對象),另一個隱藏參數是_cmd,代表當前類方法的selector。

    super並不是隱藏的參數,它只是一個"編譯器指示符"

 

2, [ super init]     都做了什麼

發送消息時

複製代碼
Class  A
-reposition { ... [self setOrigin:someX :someY]; ... }
複製代碼

 A  a= [a .. init];

[a  reposition];   方法體中  編譯器將

  [self setOrigin:someX :someY];  

 其轉換爲

objc_msgSend(id selfSEL _cmd, ...) 。self -> a

 此時 self 指代a 對象,方法從a 對應 類結構的 方法調度表中開始尋找,如果找不到,延繼承鏈往 父類中尋找 。

同樣如果 reposition 是類方法, self 指代 A 類對象。

複製代碼
Class  A
-reposition  
{  
     ...  
     [super setOrigin:someX :someY];  
     ...  
}
複製代碼

[a  reposition];   方法體中編譯器將

[super setOrigin:someX :someY]; 

其轉換爲

id objc_msgSendSuper(struct objc_super *super, SEL op, ...)

 第一個參數是個objc_super的結構體,第二個參數還是類似上面的類方法的selector,先看下objc_super這個結構體是什麼東西:

struct objc_super {
    id receiver;
   Class superClass;
};

    可以看到這個結構體包含了兩個成員,一個是 receiver,這個類似上面 objc_msgSend 的第一個參數 receiver,第二個成員是記錄寫 super 這個類的父類是什麼,拿上面的代碼爲例,當編譯器遇到 A 裏

[super setOrigin:someX :someY]

 時,開始做這幾個事:

      >構建 objc_super 的結構體,此時這個結構體的第一個成員變量 receiver 就是 a,和 self 相同。而第二個成員變量 superClass 就是指類 A的 superClass。

      >調用 objc_msgSendSuper 的方法,將這個結構體和

setOrigin

 的 sel 傳遞過去。函數裏面在做的事情類似這樣:從 objc_super 結構體指向的 superClass 的方法列表開始找 setOrigin 的 selector,找到後再以 objc_super->receiver 去調用這個 selector,可能也會使用 objc_msgSend 這個函數,不過此時的第一個參數 theReceiver 就是 objc_super->receiver,第二個參數是從 objc_super->superClass 中找到的 selector

 

 

3,爲什麼要 self =  [super init];

符合oc 繼承類 初始化規範 super 同樣也是這樣,  [super init]  去self 的super 中調用init     super 調用 superSuper 的init 。直到根類 NSObject 中的init ,

根類中init 負責初始化 內存區域  向裏面添加 一些必要的屬性,返回內存指針,  這樣 延着繼承鏈 初始化的內存指針 被從上 到 下 傳遞,在不同的子類中向塊內存添加 子類必要的屬性,直到 我們的 A 類中 得到內存指針,賦值給slef 參數, 在if (slef){//添加A 的屬性 }

 

 

 

下面來看看這個:

複製代碼
@implementation Son : Father
- (id)init
{
    self = [super init];
    if (self)
    {
        NSLog(@"%@", NSStringFromClass([self class]));
        NSLog(@"%@", NSStringFromClass([super class]));
    }
    return self;
}
@end
複製代碼

 

應該不難分析出  打印結果:

Son
Son





@implementation Son : Father
- (id)init
{
    self = [super init];
    if (self)
    {
       
    }
    return self;
}
複製代碼
發佈了18 篇原創文章 · 獲贊 1 · 訪問量 15萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章