一、什么是instancetype
instancetype是clang 3.5開始,clang提供的一個(gè)關(guān)鍵字,表示某個(gè)方法返回的未知類型的Objective-C對(duì)象。我們都知道未知類型的的對(duì)象可以用id關(guān)鍵字表示,那為什么還會(huì)再有一個(gè)instancetype呢?
二、關(guān)聯(lián)返回類型(related result types)
根據(jù)Cocoa的命名規(guī)則,滿足下述規(guī)則的方法:
1、類方法中,以alloc或new開頭
2、實(shí)例方法中,以autorelease,init,retain或self開頭
會(huì)返回一個(gè)方法所在類類型的對(duì)象,這些方法就被稱為是關(guān)聯(lián)返回類型的方法。換句話說,這些方法的返回結(jié)果以方法所在的類為類型,說的有點(diǎn)繞口,請(qǐng)看下面的例子:
@interfaceNSObject
+?(id)alloc;
-?(id)init;
@end
@interfaceNSArray?:?NSObject
@end
當(dāng)我們使用如下方式初始化NSArray時(shí):
NSArray*array?=?[[NSArrayalloc]init];
按照Cocoa的命名規(guī)則,語句[NSArrayalloc]的類型就是NSArray*因?yàn)閍lloc的返回類型屬于關(guān)聯(lián)返回類型。同樣,[[NSArrayalloc]init]的返回結(jié)果也是NSArray*。
如果一個(gè)不是關(guān)聯(lián)返回類型的方法,如下:
@interfaceNSArray
+?(id)constructAnArray;
@end
當(dāng)我們使用如下方式初始化NSArray時(shí):
[NSArray constructAnArray];
根據(jù)Cocoa的方法命名規(guī)范,得到的返回類型就和方法聲明的返回類型一樣,是id。
但是如果使用instancetype作為返回類型,如下:
@interfaceNSArray
+?(instancetype)constructAnArray;
@end
當(dāng)使用相同方式初始化NSArray時(shí):
[NSArray constructAnArray];
得到的返回類型和方法所在類的類型相同,是NSArray*!
總結(jié)一下,instancetype的作用,就是使那些非關(guān)聯(lián)返回類型的方法返回所在類的類型!
能夠確定對(duì)象的類型,能夠幫助編譯器更好的為我們定位代碼書寫問題,比如:
[[[NSArray alloc]init]mediaPlaybackAllowsAirPlay];
//??"No?visible?@interface?for?`NSArray`?declares?the?selector?`mediaPlaybackAllowsAirPlay`"
[[NSArray array]mediaPlaybackAllowsAirPlay];//?(No?error)
上例中第一行代碼,由于[[NSArrayalloc]init]的結(jié)果是NSArray*,這樣編譯器就能夠根據(jù)返回的數(shù)據(jù)類型檢測(cè)出NSArray是否實(shí)現(xiàn)mediaPlaybackAllowsAirPlay方法。有利于開發(fā)者在編譯階段發(fā)現(xiàn)錯(cuò)誤。
第二行代碼,由于array不屬于關(guān)聯(lián)返回類型方法,[NSArrayarray]返回的是id類型,編譯器不知道id類型的對(duì)象是否實(shí)現(xiàn)了mediaPlaybackAllowsAirPlay方法,也就不能夠替開發(fā)者及時(shí)發(fā)現(xiàn)錯(cuò)誤。
都可以作為方法的返回類型
①instancetype可以返回和方法所在類相同類型的對(duì)象,id只能返回未知類型的對(duì)象;
②instancetype只能作為返回值,不能像id那樣作為參數(shù),比如下面的寫法:
//err,expected?a?type
-?(void)setValue:(instancetype)value
{
//do?something
}
就是錯(cuò)的,應(yīng)該寫成:
-?(void)setValue:(id)value
{
//do?something
}