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

  1. @interface NSObject  

  2. + (id)alloc;  

  3. - (id)init;  

  4. @end  

  5.   

  6. @interface NSArray : NSObject  

  7. @end  

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

[objc] view plaincopy在CODE上查看代碼片派生到我的代碼片

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

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


三、instancetype作用

1、作用

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

[objc] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. @interface NSArray  

  2. + (id)constructAnArray;  

  3. @end  


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

[objc] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. [NSArray constructAnArray];  

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


但是如果使用instancetype作爲返回類型,如下:

[objc] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. @interface NSArray  

  2. + (instancetype)constructAnArray;  

  3. @end  

當使用相同方式初始化NSArray時:

[objc] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. [NSArray constructAnArray];  

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


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

2、好處

能夠確定對象的類型,能夠幫助編譯器更好的爲我們定位代碼書寫問題,比如:

[objc] view plaincopy在CODE上查看代碼片派生到我的代碼片

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

  1. //err,expected a type  

  2. - (void)setValue:(instancetype)value  

  3. {  

  4.     //do something  

  5. }  

就是錯的,應該寫成:

[objc] view plaincopy在CODE上查看代碼片派生到我的代碼片

  1. - (void)setValue:(id)value  

  2. {  

  3.     //do something  

  4. }  




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