1. 程式人生 > >[轉]Objective-C中的instancetype和id關鍵字

[轉]Objective-C中的instancetype和id關鍵字

一、什麼是instancetype

instancetype是clang 3.5開始,clang提供的一個關鍵字,表示某個方法返回的未知型別的Objective-C物件。我們都知道未知型別的的物件可以用id關鍵字表示,那為什麼還會再有一個instancetype呢?

二、關聯返回型別(related result types)

根據Cocoa的命名規則,滿足下述規則的方法:

1、類方法中,以alloc或new開頭

2、例項方法中,以autorelease,init,retain或self開頭

會返回一個方法所在類型別的物件,這些方法就被稱為是關聯返回型別的方法。換句話說,這些方法的返回結果以方法所在的類為型別,說的有點繞口,請看下面的例子:

[objc]  view plain  copy
  1. @interface NSObject  
  2. + (id)alloc;  
  3. - (id)init;  
  4. @end  
  5.   
  6. @interface NSArray : NSObject  
  7. @end  
當我們使用如下方式初始化NSArray時: [objc]  view plain  copy
  1. NSArray *array = [[NSArray alloc] init];  
按照Cocoa的命名規則,語句 [NSArray alloc]  的型別就是NSArray* 因為alloc的返回型別屬於關聯返回型別。同樣,[[NSArray alloc]
init]
 
的返回結果也是 NSArray*

三、instancetype作用

1、作用

如果一個不是關聯返回型別的方法,如下:

[objc]  view plain  copy
  1. @interface NSArray  
  2. + (id)constructAnArray;  
  3. @end  

當我們使用如下方式初始化NSArray時:

[objc]  view plain  copy
  1. [NSArray constructAnArray];  
根據Cocoa的方法命名規範, 得到的返回型別就和方法宣告的返回型別一樣,是id

但是如果使用instancetype作為返回型別,如下:

[objc]  view plain  copy
  1. @interface NSArray  
  2. + (instancetype)constructAnArray;  
  3. @end  
當使用相同方式初始化NSArray時: [objc]  view plain  copy
  1. [NSArray constructAnArray];  
得到的返回型別和方法所在類的型別相同, NSArray* !

總結一下,instancetype的作用,就是使那些非關聯返回型別的方法返回所在類的型別

2、好處

能夠確定物件的型別,能夠幫助編譯器更好的為我們定位程式碼書寫問題,比如:

[objc]  view plain  copy
  1. [[[NSArray alloc] init] mediaPlaybackAllowsAirPlay]; //  "No visible @interface for `NSArray` declares the selector `mediaPlaybackAllowsAirPlay`"  
  2.   
  3. [[NSArray array] mediaPlaybackAllowsAirPlay]; // (No error)  
上例中第一行程式碼,由於 [[NSArray   alloc] init] 的結果是 NSArray* ,這樣編譯器就能夠根據返回的資料型別檢測出NSArray是否實現mediaPlaybackAllowsAirPlay方法。有利於開發者在編譯階段發現錯誤。

第二行程式碼,由於array不屬於關聯返回型別方法,[NSArray array]返回的是id型別,編譯器不知道id型別的物件是否實現了mediaPlaybackAllowsAirPlay方法,也就不能夠替開發者及時發現錯誤。

四、instancetype和id的異同

1、相同點

都可以作為方法的返回型別

2、不同點

①instancetype可以返回和方法所在類相同型別的物件,id只能返回未知型別的物件;

②instancetype只能作為返回值,不能像id那樣作為引數,比如下面的寫法:

[objc]  view plain  copy
  1. //err,expected a type  
  2. - (void)setValue:(instancetype)value  
  3. {  
  4.     //do something  
  5. }  
就是錯的,應該寫成: [objc]  view plain  copy
  1. - (void)setValue:(id)value  
  2. {  
  3.     //do something  
  4. }  

五、參考

1、http://nshipster.com/instancetype/

2、http://clang.llvm.org/docs/LanguageExtensions.html#objective-c-features


————————————————————————————

作者:wangzz 原文地址:http://blog.csdn.net/wzzvictory/article/details/16994913