一句話總結:id可以作爲參數也可以返回未知類型的對象,instancetype只可以返回和方法所在類相同類型的對象(非關聯返回類型返回方法所在類的類型),id可以作爲返回值還可以作爲參數,instancetype只能作爲返回值
下面細說
一、什麼是instancetype
instancetype是clang 3.5開始,clang提供的一個關鍵字,表示某個方法返回的未知類型的Objective-C對象。我們都知道未知類型的的對象可以用id關鍵字表示,那爲什麼還會再有一個instancetype呢?
關聯返回類型(related result types)
根據Cocoa的命名規則,滿足下述規則的方法:
1、類方法中,以alloc或new開頭
2、實例方法中,以autorelease,init,retain或self開頭
會返回一個方法所在類類型的對象,這些方法就被稱爲是關聯返回類型的方法。換句話說,這些方法的返回結果以方法所在的類爲類型,說的有點繞口,請看下面的例子:
@interface NSObject
+ (id)alloc;
- (id)init;
@end
@interface NSArray : NSObject
@end
當我們使用如下方式初始化NSArray時:
NSArray *array = [[NSArray alloc] init];
按照Cocoa的命名規則,語句[NSArray alloc] 的類型就是NSArray*因爲alloc的返回類型屬於關聯返回類型。同樣,[[NSArray alloc]init] 的返回結果也是NSArray*。
二、instancetype作用
1、作用
如果一個不是關聯返回類型的方法,如下:
[objc] view plaincopy在CODE上查看代碼片派生到我的代碼片
@interface NSArray
+ (id)constructAnArray;
@end
當我們使用如下方式初始化NSArray時:
[NSArray constructAnArray];
根據Cocoa的方法命名規範,得到的返回類型就和方法聲明的返回類型一樣,是id。
但是如果使用instancetype作爲返回類型,如下:
@interface NSArray
+ (instancetype)constructAnArray;
@end
當使用相同方式初始化NSArray時:
[NSArray constructAnArray];
得到的返回類型和方法所在類的類型相同,是NSArray*!
總結一下,instancetype的作用,就是使那些非關聯返回類型的方法返回所在類的類型!
2、好處
能夠確定對象的類型,能夠幫助編譯器更好的爲我們定位代碼書寫問題,比如:
[[[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的異同
①instancetype可以返回和方法所在類相同類型的對象(非關聯返回類型),id只能返回未知類型的對象;
②instancetype只能作爲返回值,id可以作爲返回值還可以作爲參數
對的:
- (void)setValue:(id)value
{
//do something
}
錯的:
//err,expected a type
- (void)setValue:(instancetype)value
{
//do something
}
Xcode5之前 用id (id:可以調用任何對象方法,缺點是不能進行編譯檢查)
Xcode5出了instancetype:自動識別當前類的對象(哪個類調用識別成哪個類的對象,就只能調用這個類的方法)
__kindof Person * 而加上__kindof 就是代表可以是當前類及其子類可以調用這個方法