偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Objective-C 中Self 和 Super 詳解

移動(dòng)開(kāi)發(fā) iOS
本文介紹的是Objective-C 中Self 和 Super 詳解,不多說(shuō),我們先來(lái)看內(nèi)容。

Objective-CSelf Super 詳解本文要介紹的內(nèi)容,在 Objective-C 中的類實(shí)現(xiàn)中經(jīng)常看到這兩個(gè)關(guān)鍵字 self super,以以前 oop 語(yǔ)言的經(jīng)驗(yàn),拿 c++ 為例,self 相當(dāng)于 this,super 相當(dāng)于調(diào)用父類的方法,這么看起來(lái)是很容易理解的。以下面的

在 Objective-C 中的類實(shí)現(xiàn)中經(jīng)??吹竭@兩個(gè)關(guān)鍵字 ”self” 和 ”super”,以以前 oop 語(yǔ)言的經(jīng)驗(yàn),拿 c++ 為例,self 相當(dāng)于 this,super 相當(dāng)于調(diào)用父類的方法,這么看起來(lái)是很容易理解的。以下面的代碼為例:

  1. @interface Person:NSObject {  
  2.     NSString*  name;  
  3. }  
  4. - (void) setName:(NSString*) yourName;  
  5. @end  
  6.  
  7. @interface PersonMe:Person {  
  8.     NSUInteger age;  
  9. }  
  10. - (void) setAge:(NSUInteger) age;  
  11. - (void) setName:(NSString*) yourName andAge:(NSUInteger) age;  
  12. @end  
  13.  
  14. @implementation PersonMe  
  15. - (void) setName:(NSString*) yourName andAge:(NSUInteger) age {  
  16.     [self setAge:age];  
  17.     [super setName:yourName];  
  18. }  
  19. @end  
  20.  
  21. int main(int argc, char* argv[]) {  
  22.     NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init]  
  23.     PersonMe* me = [[PersonMe alloc] init];  
  24.     [me setName:@"asdf" andAge:18];  
  25.     [me release];  
  26.     [pool drain];  
  27.     return 0;  

上面有簡(jiǎn)單的兩個(gè)類,在子類PersonMe中調(diào)用了自己類中的setAge和父類中的setName,這些代碼看起來(lái)很好理解,沒(méi)什么問(wèn)題。
然后我在setName:andAge的方法中加入兩行:

  1. NSLog(@"self ' class is %@", [self class]);  
  2. NSLog(@"super' class is %@", [super class]); 

這樣在調(diào)用時(shí),會(huì)打出來(lái)這兩個(gè)的class,先猜下吧,會(huì)打印出什么?按照以前oop語(yǔ)言的經(jīng)驗(yàn),這里應(yīng)該會(huì)輸出:

  1. self ' s class is PersonMe  
  2. super ' s class is Person 

但是編譯運(yùn)行后,可以發(fā)現(xiàn)結(jié)果是:

  1. self 's class is PersonMe  
  2. super ' s class is PersonMe 

self 的 class 和預(yù)想的一樣,怎么 super 的 class 也是 PersonMe?

真相

self 是類的隱藏的參數(shù),指向當(dāng)前當(dāng)前調(diào)用方法的類,另一個(gè)隱藏參數(shù)是 _cmd,代表當(dāng)前類方法的 selector。這里只關(guān)注這個(gè) self。super 是個(gè)啥?super 并不是隱藏的參數(shù),它只是一個(gè)“編譯器指示符”,它和 self 指向的是相同的消息接收者,拿上面的代碼為例,不論是用 [self setName] 還是 [super setName],接收“setName”這個(gè)消息的接收者都是 PersonMe* me 這個(gè)對(duì)象。不同的是,super 告訴編譯器,當(dāng)調(diào)用 setName 的方法時(shí),要去調(diào)用父類的方法,而不是本類里的。

當(dāng)使用 self 調(diào)用方法時(shí),會(huì)從當(dāng)前類的方法列表中開(kāi)始找,如果沒(méi)有,就從父類中再找;而當(dāng)使用 super 時(shí),則從父類的方法列表中開(kāi)始找。然后調(diào)用父類的這個(gè)方法

One more step

這種機(jī)制到底底層是如何實(shí)現(xiàn)的?其實(shí)當(dāng)調(diào)用類方法的時(shí)候,編譯器會(huì)將方法調(diào)用轉(zhuǎn)成一個(gè) C 函數(shù)方法調(diào)用,Apple 的 objcRuntimeRef 上說(shuō):

Sending Messages

When it encounters a method invocation, the compiler might generate a call to any of several functions to perform the actual message dispatch, depending on the receiver, the return value, and the arguments. You can use these functions to dynamically invoke methods from your own plain C code, or to use argument forms not permitted by NSObject’s perform… methods. These functions are declared in /usr/include/objc/objc-runtime.h.
   
objc_msgSend sends a message with a simple return value to an instance of a class.

objc_msgSend_stret sends a message with a data-structure return value to an instance of

a class.

objc_msgSendSuper sends a message with a simple return value to the superclass of an instance of a class.

objc_msgSendSuper_stret sends a message with a data-structure return value to the superclass of an instance of a class.

可以看到會(huì)轉(zhuǎn)成調(diào)用上面 4 個(gè)方法中的一個(gè),由于 _stret 系列的和沒(méi)有 _stret 的那兩個(gè)類似,先只關(guān)注 objc_msgSend 和 objc_msgSendSuper 兩個(gè)方法。當(dāng)使用 [self setName] 調(diào)用時(shí),會(huì)使用 objc_msgSend 的函數(shù),先看下 objc_msgSend 的函數(shù)定義:

  1. id objc_msgSend(id theReceiver, SEL theSelector, ...) 

第一個(gè)參數(shù)是消息接收者,第二個(gè)參數(shù)是調(diào)用的具體類方法的 selector,后面是 selector 方法的可變參數(shù)。我們先不管這個(gè)可變參數(shù),以 [self setName:] 為例,編譯器會(huì)替換成調(diào)用 objc_msgSend 的函數(shù)調(diào)用,其中 theReceiver 是 self,theSelector 是 @selector(setName:),這個(gè) selector 是從當(dāng)前 self 的 class 的方法列表開(kāi)始找的 setName,當(dāng)找到后把對(duì)應(yīng)的 selector 傳遞過(guò)去。

而當(dāng)使用 [super setName] 調(diào)用時(shí),會(huì)使用 objc_msgSendSuper 函數(shù),看下 objc_msgSendSuper 的函數(shù)定義:

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

第一個(gè)參數(shù)是個(gè)objc_super的結(jié)構(gòu)體,第二個(gè)參數(shù)還是類似上面的類方法的selector,先看下objc_super這個(gè)結(jié)構(gòu)體是什么東西:

  1. struct objc_super {  
  2.     id receiver;  
  3.    Class superClass;  
  4. }; 

可以看到這個(gè)結(jié)構(gòu)體包含了兩個(gè)成員,一個(gè)是 receiver,這個(gè)類似上面 objc_msgSend 的第一個(gè)參數(shù) receiver,第二個(gè)成員是記錄寫(xiě) super 這個(gè)類的父類是什么,拿上面的代碼為例,當(dāng)編譯器遇到 PersonMe 里 setName:andAge 方法里的 [super setName:] 時(shí),開(kāi)始做這幾個(gè)事:

構(gòu)建 objc_super 的結(jié)構(gòu)體,此時(shí)這個(gè)結(jié)構(gòu)體的第一個(gè)成員變量 receiver 就是 PersonMe* me,和 self 相同。而第二個(gè)成員變量 superClass 就是指類 Person,因?yàn)?PersonMe 的超類就是這個(gè) Person。

調(diào)用 objc_msgSendSuper 的方法,將這個(gè)結(jié)構(gòu)體和 setName 的 sel 傳遞過(guò)去。函數(shù)里面在做的事情類似這樣:從 objc_super 結(jié)構(gòu)體指向的 superClass 的方法列表開(kāi)始找 setName 的 selector,找到后再以 objc_super->receiver 去調(diào)用這個(gè) selector,可能也會(huì)使用 objc_msgSend 這個(gè)函數(shù),不過(guò)此時(shí)的第一個(gè)參數(shù) theReceiver 就是 objc_super->receiver,第二個(gè)參數(shù)是從 objc_super->superClass 中找到的 selector

里面的調(diào)用機(jī)制大體就是這樣了,以上面的分析,回過(guò)頭來(lái)看開(kāi)始的代碼,當(dāng)輸出 [self class] 和 [super class] 時(shí),是個(gè)怎樣的過(guò)程。

當(dāng)使用 [self class] 時(shí),這時(shí)的 self 是 PersonMe,在使用 objc_msgSend 時(shí),第一個(gè)參數(shù)是 receiver 也就是 self,也是 PersonMe* me 這個(gè)實(shí)例。第二個(gè)參數(shù),要先找到 class 這個(gè)方法的 selector,先從 PersonMe 這個(gè)類開(kāi)始找,沒(méi)有,然后到 PersonMe 的父類 Person 中去找,也沒(méi)有,再去 Person 的父類 NSObject 去找,一層一層向上找之后,在 NSObject 的類中發(fā)現(xiàn)這個(gè) class 方法,而 NSObject 的這個(gè) class 方法,就是返回 receiver 的類別,所以這里輸出 PersonMe。

當(dāng)使用 [super class] 時(shí),這時(shí)要轉(zhuǎn)換成 objc_msgSendSuper 的方法。先構(gòu)造 objc_super 的結(jié)構(gòu)體吧,第一個(gè)成員變量就是 self,第二個(gè)成員變量是 Person,然后要找 class 這個(gè) selector,先去 superClass 也就是 Person 中去找,沒(méi)有,然后去 Person 的父類中去找,結(jié)果還是在 NSObject 中找到了。然后內(nèi)部使用函數(shù) objc_msgSend(objc_super->receiver, @selector(class))  去調(diào)用,此時(shí)已經(jīng)和我們用 [self class] 調(diào)用時(shí)相同了,此時(shí)的 receiver 還是 PersonMe* me,所以這里返回的也是 PersonMe。

小結(jié):關(guān)于Objective-CSelf Super 詳解的內(nèi)容介紹完了希望本文對(duì)你有所幫助!

責(zé)任編輯:zhaolei 來(lái)源: 博客園
相關(guān)推薦

2011-07-20 13:34:37

Objective-C self.

2011-07-29 16:16:30

Objective-c block

2011-08-17 10:58:59

Objective-C構(gòu)造函數(shù)

2011-07-29 15:47:21

iPhone開(kāi)發(fā) Objective- C

2015-10-08 10:01:10

Objective-CLayout

2011-07-18 16:36:51

Objective-C XCode

2014-04-01 10:50:42

iOS開(kāi)發(fā)runtimeObjective-C

2011-08-17 10:29:39

Objective-C預(yù)處理

2011-07-06 11:19:45

Objective-C

2011-08-04 09:56:30

Objective-C 變量 數(shù)據(jù)類型

2012-01-18 10:13:50

Objective-CiOSself

2011-08-04 13:38:01

Objective-C C++

2011-07-27 16:55:12

Objective-c 閉包

2011-08-01 17:11:43

Objective-C 函數(shù)

2011-08-15 14:32:42

Objective-C委托協(xié)議

2014-04-28 09:56:56

Objective-CiOS命名空間

2011-08-17 11:05:22

Objective-C方法

2011-08-16 13:43:40

Objective-C文件cocoa

2011-08-16 10:23:04

Objective-CNSAutoreleaXcode常用鍵

2013-05-02 10:51:17

iOS開(kāi)發(fā)Objective-C@property
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)