OC:關于self和super 在Objective中的疑惑 與 分析??

  • 版權聲明:本文為博主原創文章,未經博主允許不得轉載。

這個問題貌似很初級,但很容易讓人忽略,me too 。

面試一定都是很注重基礎的,不管高級還是初級。

雖然基礎好跟基礎不好都可以寫代碼,網上那么多資料。 區分高低也就是研究的深度和廣度。

開始我們的問題:

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

這段代碼估計很多人都寫爛了,就算沒寫爛,Xcode自動生成的我們也看吐了。 好吧,來說說原來.

上來就是 : 這個其實就是在子類實現初始化前調用父類的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 self,SEL _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

當 發送 class 消息 時不管是 self 還是 super 其消息主體依然是 self ,也就是說 selfsuper 指向的 是同一個對象。只是 查找方法的位置 區別,一個從本類,一個從本類的超類
一般情況下 class 方法 只有在 根類 NSObject 中定義,極少情況有子類重寫 class 方法,
所以 [slef class][super class] 都是在 根類中 找方法實現, 消息接收主體 又都是 a
如果重寫可能會不一樣。
自然都打印出 Son.

  • 在來一個例子:
#import <Foundation/Foundation.h>

@interface EngineSuper : NSObject
-(void)printCurrentClass;

@end

#import "EngineSuper.h"

@implementation EngineSuper
-(void)printCurrentClass{

    NSLog(@"EngineSuper=======%@",[self class]);
}
@end



@interface Engine : EngineSuper
-(void)printSupClass;
@end


@implementation Engine

-(void)printSupClass{
   [super printCurrentClass];
}

//調用:
Engine *engine = [[Engine alloc]init];
[engine  printCurrentClass];//直接調用父類 方法,engine沒重載 它

[engine  printSupClass];//間接調用父類方法,
  • 打印當然都是 :
Engine
Engine```

>方法體中` self` 始終指代 方法的接收者 及對象  engine。,
 換成   NSLog(@"EngineSuper=======%@",[super class]); 結果也是`一樣的`。


>`super` 就是個障眼法 發,`編譯器符號`, 它可以替換成 `[slef class]`,只不過 方法是從 self 的`超類`開始 尋找。
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,443評論 6 532
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,530評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,407評論 0 375
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,981評論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,759評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,204評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,263評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,415評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,955評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,782評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,983評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,528評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,222評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,650評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,892評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,675評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,967評論 2 374

推薦閱讀更多精彩內容

  • 轉至元數據結尾創建: 董瀟偉,最新修改于: 十二月 23, 2016 轉至元數據起始第一章:isa和Class一....
    40c0490e5268閱讀 1,751評論 0 9
  • 我們常常會聽說 Objective-C 是一門動態語言,那么這個「動態」表現在哪呢?我想最主要的表現就是 Obje...
    Ethan_Struggle閱讀 2,225評論 0 7
  • super和self相信大家每天都在和它們打交道,可能很多人都是按照套路來寫,沒怎么仔細思考過它們之間的關系,之...
    舒城8中閱讀 635評論 0 1
  • Kotlin是JetBrains開發的基于JVM的語言,JetBrains想必大家應該很熟悉了,他們創造了很多強大...
    韶指年華閱讀 504評論 0 0
  • 時間在過,年齡在長,心也在慢慢成熟。 以前總害怕一個人,可又不愿意拉下臉來和別人說:你去哪?帶上我吧。總是制造出:...
    走阿走阿走閱讀 248評論 0 2