第14条:理解“类对象”的用意
對象類型并非在編譯期就綁定好了,而是要在運(yùn)行期查找。
?
id 類型,它能指代任意的 OC 對象類型。編譯器假定它能響應(yīng)所有消息。
對象類型 id,由于其本身已經(jīng)是指針了。
?
在運(yùn)行期檢視對象類型,這一操作也叫做“類型信息查詢”(introspection, “內(nèi)省”),
這個(gè)特性內(nèi)置于 Foundation 框架的 NSObject 協(xié)議里,凡是由公共根類(即 NSObjcet 與 NSProxy)繼承而來的對象都要遵從此協(xié)議。
?
OC 對象的本質(zhì):每一個(gè) OC 對象實(shí)例都是指向某塊內(nèi)存數(shù)據(jù)的指針。
?
描述 OC 對象所用的數(shù)據(jù)結(jié)構(gòu)定義在運(yùn)行期程序庫的頭文件里,id 類型本身也在定義在這里:
typedef struct objc_object {
Class isa; ? // 定義了對象所屬的類,通常稱為“is a”指針。
} *id;
?
Class 對象也定義在運(yùn)行期程序庫的頭文件中:
此結(jié)構(gòu)體存放類的“元數(shù)據(jù)”(metadata)。
typedef struct objc_class *Class;
struct objc_class {
Class isa; ? ?//Class ?本身亦為 OC 對象。 //isa 指針?biāo)赶虻念愋褪橇硗庖粋€(gè)類,叫做“元類”(metaclass),用來表述類對象本身所具備的元數(shù)據(jù)。
Class super_class; ? ? //本類的超類
const char *name;
long version;
long info;
long instance_size;
struct objc_ivar_list *ivars;
struct objc_method_list ?**methodLists;
struct objc_cache *cache;
struct objc)protocol_list *protocols;
};
?
如果對象類型無法在編譯期確定,那么就應(yīng)該使用類型信息查詢方法來探知。
用類型信息查詢方法來檢視類繼承體系:(由于對象是動(dòng)態(tài)的,這特性極為重要)
“isMemberOfClass:” ? ?判斷出對象是否為某個(gè)特定類的實(shí)例,
“isKindOfClass:” ? ? ? ? 判斷出對象是否為某類或派生類的實(shí)例。?
?
比較類對象是否等同的辦法:
使用== 操作符,而不要使用 isEqual:方法
原因在于,類對象是“單例”(在應(yīng)用程序范圍內(nèi),每個(gè)類的 Class 僅有一個(gè)實(shí)例)。
?
某個(gè)對象可能會(huì)把其收到的所有方法都轉(zhuǎn)發(fā)給另外一個(gè)對象。這樣的對象叫做“代理”,此種對象均以 NSProxy 為根類。
通常情況下,如果在此種代理對象上調(diào)用 class 方法,那么返回的是代理對象本身(此類是 NSProxy 的子類),而非接受的代理的對象所屬的類。然而,若改用“iKindOfClass:”這樣的類型信息查詢方法,那么代理對象就會(huì)把這條消息轉(zhuǎn)給“接受代理的對象”。
也就是說,這條消息的返回值與直接在接受代理的對象上面查詢其類型所得的結(jié)果相同。因此,這樣查出來的類對象與通過 calss 方法所返回的那個(gè)類對象不同,class 方法所返回的“類表示發(fā)起代理的對象,而非接受代理的對象”。
所以盡量使用類型信息查詢方法來確定對象類型,而不要直接比較類對象,因?yàn)槟承ο罂赡軐?shí)現(xiàn)了消息轉(zhuǎn)發(fā)功能。
?
?
轉(zhuǎn)載于:https://www.cnblogs.com/Pikdays/p/4371471.html
與50位技術(shù)專家面對面20年技術(shù)見證,附贈(zèng)技術(shù)全景圖總結(jié)
以上是生活随笔為你收集整理的第14条:理解“类对象”的用意的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 奇怪吸引子---Chua
- 下一篇: CentOS 7.0安装Nvidia驱动