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
在CODE上查看代碼片派生到我的代碼片

@interface NSObject  
+ (id)alloc;  
- (id)init;  
@end  

@interface NSArray : NSObject  
@end  

當我們使用如下方式初始化NSArray時:
[objc] view plain copy
在CODE上查看代碼片派生到我的代碼片

NSArray *array = [[NSArray alloc] init];  

按照Cocoa的命名規則,語句[NSArray alloc] 的類型就是NSArray*因爲alloc的返回類型屬於關聯返回類型。同樣,[[NSArray alloc]init] 的返回結果也是NSArray*。

三、instancetype作用
1、作用

如果一個不是關聯返回類型的方法,如下:
[objc] view plain copy
在CODE上查看代碼片派生到我的代碼片

@interface NSArray  
+ (id)constructAnArray;  
@end  

當我們使用如下方式初始化NSArray時:
[objc] view plain copy
在CODE上查看代碼片派生到我的代碼片

[NSArray constructAnArray];  

根據Cocoa的方法命名規範,得到的返回類型就和方法聲明的返回類型一樣,是id。

但是如果使用instancetype作爲返回類型,如下:
[objc] view plain copy
在CODE上查看代碼片派生到我的代碼片

@interface NSArray  
+ (instancetype)constructAnArray;  
@end  

當使用相同方式初始化NSArray時:
[objc] view plain copy
在CODE上查看代碼片派生到我的代碼片

[NSArray constructAnArray];  

得到的返回類型和方法所在類的類型相同,是NSArray*!

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

能夠確定對象的類型,能夠幫助編譯器更好的爲我們定位代碼書寫問題,比如:
[objc] view plain copy
在CODE上查看代碼片派生到我的代碼片

[[[NSArray alloc] init] mediaPlaybackAllowsAirPlay]; //  "No visible @interface for `NSArray` declares the selector `mediaPlaybackAllowsAirPlay`"  

[[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
在CODE上查看代碼片派生到我的代碼片

//err,expected a type  
- (void)setValue:(instancetype)value  
{  
    //do something  
}  

就是錯的,應該寫成:
[objc] view plain copy
在CODE上查看代碼片派生到我的代碼片

- (void)setValue:(id)value  
{  
    //do something  
}  

五、參考

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

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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章