成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Cocoa 基本原理指南 內省介紹

移動開發 iOS
內省(Introspection)是面向對象語言和環境的一個強大特性,Objective-C和Cocoa在這個方面尤其的豐富。

Cocoa 基本原理指南 內省介紹是本文介紹的內容,內省(Introspection)是面向對象語言和環境的一個強大特性,Objective-C和Cocoa在這個方面尤其的豐富。內省是對象揭示自己作為一個運行時對象的詳細信息的一種能力。這些詳細信息包括對象在繼承樹上的位置,對象是否遵循特定的協議,以及是否可以響應特定的消息。NSObject協議和類定義了很多內省方法,用于查詢運行時信息,以便根據對象的特征進行識別。

明智地使用內省可以使面向對象的程序更加高效和強壯。它有助于避免錯誤地進行消息派發、錯誤地假設對象相等、以及類似的問題。下面的部分將介紹如何在代碼中有效地使用NSObject的內省方法。

評估繼承關系

一旦您知道一個對象屬于什么類,就可能已經相當了解這個對象了。您可以知道它具有什么能力、哪些屬性、以及可以響應哪些消息。即使在內省之后不能了解對象所屬的類,也可以知道該對象不能響應特定的消息。

NSObject協議聲明了幾個方法,用于確定對象在類層次中的位置。這些方法在不同粒度上進行操作,比如class和superclass實例方法分別返回代表類和超類的Class對象。使用這些方法需要將一個Class對象和另一個進行對比。

使用類和超類的方法

  1. // ...   
  2. while ( id anObject = [objectEnumerator nextObject] ) {   
  3.     if ( [self class] == [anObject superclass] ) {   
  4.         // do something appropriate...   
  5.     }   
  6. }  

請注意:有些時候您需要通過class或superclass方法得到正確的類消息接收者。

更加常見的是檢查對象類的從屬關系,這種情況下您需要向該對象發送isKindOfClass:或isMemberOfClass:消息。前一個方法返回接收者是否為給定類或其繼承類的實例,isMemberOfClass:消息則告訴您接收者是否為指定類的實例。isKindOfClass: 方法通常更有用,因為通過它可以知道是否可以向該對象發送一系列消息。考慮列表2-8中的代碼片斷:

 使用isKindOfClass:方法

  1. if ([item isKindOfClass:[NSData class]]) {   
  2.     const unsigned char *bytes = [item bytes];   
  3.     unsigned int length = [item length];   
  4.     // ...   
  5. }  

確定tem對象是NSData類的繼承類的實例之后,代碼就知道可以向它發送NSData的bytes和length消息。假定item是NSMutableData類的一個實例,則isKindOfClass:和isMemberOfClass:之間的差別就變得更加明顯。如果您調用的是isMemberOfClass:,而不是isKindOfClass:,條件控制塊中的代碼將永遠不會被執行,因為item并不是NSData類的實例,而是其子類NSMutableData的實例。

方法實現和協議遵循

NSObject還有兩個功能更加強大的內省方法,即respondsToSelector:和conformsToProtocol:。這兩個方法分別告訴您一個對象是否實現特定的方法,以及是否遵循指定的正式協議(即該對象是否采納了該協議,且實現了該協議的所有方法)。

在代碼中,您可以在類似的情況下使用這些方法。通過這些方法,您可以在將消息或消息集合發送給某些潛在的匿名對象之前,確定它們是否可以正確地進行響應。在發送消息之前進行檢查可以避免由不能識別的選擇器引起的運行時例外。在實現非正式協議(這種協議是委托技術的基礎)時,Application Kit就是在調用委托方法之前檢查委托對象是否實現該方法(通過respondsToSelector:方法)。

顯示了如何在代碼中使用respondsToSelector:方法。

使用respondsToSelector:方法

  1. - (void)doCommandBySelector:(SEL)aSelector {   
  2.     if ([self respondsToSelector:aSelector]) {   
  3.         [self performSelector:aSelector withObject:nil];   
  4.     } else {   
  5.         [_client doCommandBySelector:aSelector];   
  6.     }   
  7. }  

顯示如何在代碼中使用conformsToProtocol:方法:

使用conformsToProtocol:方法

  1. // ...   
  2. if (!([((id)testObject) conformsToProtocol:@protocol(NSMenuItem)])) {   
  3.     NSLog(@"Custom MenuItem, '%@', not loaded; it must conform to the   
  4.         'NSMenuItem' protocol.\n", [testObject class]);   
  5.     [testObject release];   
  6.     testObject = nil;   
  7. }  

對象的比較

hashisEqual:方法雖然不是嚴格的內省方法,但是可以發揮類似的作用,是進行對象的識別和比較時不可或缺的運行時工具。它們并不向運行環境查詢對象信息,而是依賴于具體類的比較邏輯。

hashisEqual:方法都在NSObject協議中聲明,且彼此關系緊密。實現hash方法必須返回一個整型數,作為哈希表結構中的表地址。兩個對象相等(isEqual:方法的判斷結果)意味著它們有相同的哈希值。如果您的對象可能被包含在象NSSet這樣的集合中,則需要定義hash方法,并確保該方法在兩個對象相等的時候返回相同的哈希值。NSObject類中缺省的isEqual:實現只是簡單地檢查指針是否相等。

isEqual:的使用相當直接,它將消息的接收者和通過參數傳入的對象進行比較。對象的比較常常可以在運行時決定應該對對象做些什么。如列表2-11所示,您可以通過isEqual:來確定是否執行某一個動作。在這個例子中,動作是指保存被修改了的預置信息。

使用isEqual:方法

  1. - (void)saveDefaults {   
  2.     NSDictionary *prefs = [self preferences];   
  3.     if (![origValues isEqual:prefs])    
  4.         [Preferences savePreferencesToDefaults:prefs];   
  5. }  

如果您正在創建子類,則可能需要重載isEqual:方法,以進一步檢查對象是否相等。子類可能定義額外的屬性,當兩個實例被認為相等時,屬性的值必須相同。舉例來說,假定您創建一個名為MyWidget的NSObject子類,類中包含兩個實例變量:name和data。當MyWidget的兩個實例被認為是相等時,這些變量必須具有相同的值。列表2-12顯示如何在MyWidget類中實現isEqual:方法。

重載isEqual:方法

  1. - (BOOL)isEqual:(id)other {   
  2.     if (other == self)    
  3.         return YES;   
  4.     if (!other || ![other isKindOfClass:[self class]])    
  5.         return NO;   
  6.     return [self isEqualToWidget:other];   
  7. }   
  8.     
  9. - (BOOL)isEqualToWidget:(MyWidget *)aWidget {   
  10.     if (self == aWidget)    
  11.         return YES;   
  12.     if (![(id)[self name] isEqual:[aWidget name]])   
  13.         return NO;   
  14.     if (![[self data] isEqualToData:[aWidget data]])   
  15.         return NO;   
  16.     return YES;   
  17. }  

isEqual:方法首先檢查指針的等同性,然后是類的等同性,***調用對象的比較器進行比較。比較器的名稱指示出參與比較的對象的類名稱。這種類型的比較器對傳入的對象進行強制類型檢查,是Cocoa中常見的約定,NSString的isEqualToString:和NSTimeZone的isEqualToTimeZone:就是兩個這樣的例子。特定類的比較器(在這個例子中是isEqualToWidget:)負責執行name和data變量的等同性。

Cocoa框架的所有isEqualToType:方法中,nil都不是正當的參數,這些方法的實現在接收到nil參數時會拋出例外。然而為了向后兼容,Cocoa框架中的isEqual:方法可以接收nil值,在這種情況下返回NO。

小結:Cocoa 基本原理指南 內省介紹的內容介紹完了,希望本文對你有所幫助!

責任編輯:zhaolei 來源: 互聯網
相關推薦

2011-07-07 13:30:32

Cocoa Core

2011-07-07 14:46:10

Cocoa Xcode

2011-07-07 14:22:27

Cocoa 對象 生命周期

2011-08-10 19:33:09

Cocoa對象

2010-03-18 20:13:03

Java socket

2012-01-12 14:37:34

jQuery

2011-11-29 12:17:00

2009-02-24 09:43:00

IP電話原理

2016-08-17 23:53:29

網絡爬蟲抓取系統

2021-02-08 21:40:04

SockmapBPF存儲

2019-11-28 10:45:28

ZooKeeper源碼分布式

2016-08-18 00:04:09

網絡爬蟲抓取系統服務器

2010-08-20 13:29:33

OFDM

2020-03-21 14:57:14

手機定位智能手機APP

2013-04-07 14:09:55

Android應用基本

2010-03-17 13:35:02

2009-06-11 09:56:09

MySQL Repli原理

2020-12-29 16:55:44

ZooKeeper運維數據結構

2012-09-28 10:12:55

2010-01-07 09:53:09

Winform多線程編
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久精品中文字幕 | 久久成人精品视频 | www.狠狠干| 成人精品一区二区三区 | 久久久久久成人 | 国产三级精品三级在线观看四季网 | 伊人久久精品一区二区三区 | 日本精品一区二区三区视频 | 四虎影 | 91国内精精品久久久久久婷婷 | 国产精品久久久久久久久久久久久久 | 亚洲精品在线免费播放 | 亚洲日本欧美日韩高观看 | 亚洲一区二区三区久久久 | 精品国产91乱码一区二区三区 | 羞羞视频网 | 你懂的国产 | 一区二区三区四区在线 | 天天干干 | 欧美最猛黑人 | 久久久久久色 | 精品无码久久久久久国产 | 99视频在线免费观看 | 国产日韩视频在线 | 国产精品亚洲精品 | 天堂va在线观看 | 国产视频黄色 | 国产区在线免费观看 | 91九色视频 | 久久久www成人免费无遮挡大片 | 一区二区三区中文字幕 | 最近最新中文字幕 | 国产97在线看 | 欧美激情精品久久久久 | 91九色视频在线 | 在线看亚洲 | 国产精品精品视频一区二区三区 | 亚洲一区二区黄 | 久久久久久中文字幕 | 国产精品视频一区二区三 | 亚洲国产成人在线视频 |