????? super和self相信大家每天都在和它們打交道,可能很多人都是按照套路來寫,沒怎么仔細思考過它們之間的關(guān)系,之前陸陸續(xù)續(xù)看過一些文章,介紹過它們之間的關(guān)系和區(qū)別,今天咱們決定對它們來一次徹底的摸底,看看它們在調(diào)用時究竟有什么相同點和不同點。
我們先從這幾個問題來入手。
* 什么是self
*什么是super
*[super init]做了什么
*為什么要self = [super init]
1,什么是self
? 在動態(tài)方法中,self即是“對象”
?在靜態(tài)方法中,self即是“類”
總而言之,self代表著當(dāng)前方法調(diào)用者
2,什么是super
? super 和self 都是OC提供的保留字,但兩者有本質(zhì)的區(qū)別
self是類的隱藏的參數(shù)變量,指向當(dāng)前調(diào)用方法的對象(類也是對象,類對象),另一個隱藏參數(shù)是_cmd,代表當(dāng)前類方法的selector。
super并不是隱藏的參數(shù),它只是一個"編譯器指示符"
3,[super init] 做了什么
發(fā)消息時
Class? A
-reposition{
...[self setOrigin:someX :someY];...
}
A? a= [a .. init];
[a ?reposition];?
方法體中 ?編譯器將[self setOrigin:someX :someY];
其轉(zhuǎn)換為objc_msgSend(idself,SEL_cmd,...) 。self -> a此時 self 指代a 對象,方法從a 對應(yīng) 類結(jié)構(gòu)的 方法調(diào)度表中開始尋找,如果找不到,延繼承鏈往 父類中尋找 。
同樣如果?reposition 是類方法, self 指代 A 類對象。
Class? A
-reposition{...
[super setOrigin:someX :someY];
...}
[a? reposition];? 方法體中編譯器將
[super setOrigin:someX :someY];
其轉(zhuǎn)換為
id objc_msgSendSuper(struct objc_super *super, SEL op, ...)
第一個參數(shù)是個objc_super的結(jié)構(gòu)體,第二個參數(shù)還是類似上面的類方法的selector,先看下objc_super這個結(jié)構(gòu)體是什么東西:
struct objc_super {
id receiver;
Class superClass;
};
可以看到這個結(jié)構(gòu)體包含了兩個成員,一個是 receiver,這個類似上面 objc_msgSend 的第一個參數(shù) receiver,第二個成員是記錄寫 super 這個類的父類是什么,拿上面的代碼為例,當(dāng)編譯器遇到 A 里
[super setOrigin:someX :someY]
時,開始做這幾個事:
>構(gòu)建 objc_super 的結(jié)構(gòu)體,此時這個結(jié)構(gòu)體的第一個成員變量 receiver 就是 a,和 self 相同。而第二個成員變量 superClass 就是指類 A的 superClass。
>調(diào)用 objc_msgSendSuper 的方法,將這個結(jié)構(gòu)體和
setOrigin
的 sel 傳遞過去。函數(shù)里面在做的事情類似這樣:從 objc_super 結(jié)構(gòu)體指向的 superClass 的方法列表開始找
setOrigin 的 selector,找到后再以 objc_super->receiver 去調(diào)用這個 selector,可能也會使用
objc_msgSend 這個函數(shù),不過此時的第一個參數(shù) theReceiver 就是
objc_super->receiver,第二個參數(shù)是從 objc_super->superClass 中找到的 selector
4,為什么要 self = [super init]
符合oc 繼承類 初始化規(guī)范 super 同樣也是這樣,? [super init]? 去self 的super 中調(diào)用init? ? super 調(diào)用 superSuper 的init 。直到根類 NSObject 中的init ,
根類中init 負責(zé)初始化 內(nèi)存區(qū)域 ?向里面添加 一些必要的屬性,返回內(nèi)存指針, ?這樣 延著繼承鏈 初始化的內(nèi)存指針 被從上 到 下
傳遞,在不同的子類中向塊內(nèi)存添加 子類必要的屬性,直到 我們的 A 類中 得到內(nèi)存指針,賦值給slef 參數(shù), 在if (slef){//添加A
的屬性 }
以下代碼:
@implementationSon : Father- (id)init{
self=[super init];if(self){
NSLog(@"%@", NSStringFromClass([selfclass]));
NSLog(@"%@", NSStringFromClass([superclass]));
}returnself;
}@end
應(yīng)該不難分析出? 打印結(jié)果:
Son
Son
當(dāng) 發(fā)送 class 消息 時不管是 self? 還是 super 其消息主體依然是? self ,也就是說 self 和 super 指向的 是同一個對象。只是 查找方法的位置 區(qū)別,一個從本類,一個從本類的超類。
一般情況下? class 方法 只有在 根類 NSObject 中定義,極少情況有子類重寫 class 方法,所以? [slef? class] 和 [super class] 都是在 根類中 找方法實現(xiàn), 消息接收主體 又都是 a
如果重寫可能會不一樣。自然都打印出? Son