Cocoa對象 根類
Cocoa對象 根類是本文要介紹的內(nèi)容,僅憑Objective-C語言和運(yùn)行環(huán)境并不足以構(gòu)造哪怕是最簡單的面向?qū)ο蟮某绦颍辽偈遣蝗菀椎?。還缺少一些東西:即所有對象公有的基本行為和接口的定義。根類正是提供了這些定義。
之所以叫根類,是因?yàn)樗挥谡麄€類層次(這里是指Cocoa的類層次)的根上。根類不從其它類繼承,但是類層次中的所有其它類都最終從根類繼承下來。根類連同Objective-C語言,是Cocoa直接訪問Objective-C運(yùn)行環(huán)境或與之交互的基本途徑。Cocoa對象的大部分對象行為能力都是從根類得到的。
Cocoa提供了兩個根類:NSObject和NSProxy。Cocoa將后者定義為抽象類,用于表示其它對象的替身對象。因此NSProxy類在分布式對象架構(gòu)中是很重要的。由于作用比較特別,NSProxy在Cocoa程序中出現(xiàn)頻率很低。Cocoa開發(fā)者在提到根類時,幾乎總是指NSObject。
本部分將討論NSObject類,看看它如何與運(yùn)行環(huán)境進(jìn)行交互,以及它為所有Cocoa對象定義的基本行為和接口。其中主要是它為對象的內(nèi)存分配、初始化、內(nèi)存管理、內(nèi)省、以及運(yùn)行環(huán)境支持所聲明的方法。這些概念是理解Cocoa的基礎(chǔ)。
NSObject
NSObject是大多數(shù)Objective-C類層次的根類,它沒有超類。其它類從NSObject繼承訪問Objective-C語言運(yùn)行時系統(tǒng)的基本接口,它們的實(shí)例可以得到對象行為的能力。
雖然NSObject不是一個嚴(yán)格的抽象類,但它是個虛類。僅憑一個NSObject實(shí)例除了作為一個簡單的對象外,不能完成任何有用的工作。為了在您的程序中加入特有的屬性和邏輯,必須創(chuàng)建一個或多個從NSObject或其派生類繼承下來的類。
NSObject采納了NSObject協(xié)議(參見"根類—和協(xié)議"部分)。NSObject協(xié)議支持多個根對象。舉例來說,NSProxy是另一個根類,它不是繼承自NSObject,但采納了NSObject協(xié)議,以便和其它Objective-C對象共用一個公共的接口。
NSObject和java.lang.Object一起,是Java版本的Cocoa中所有類的根類,包括Foundation和Application Kit。
根類—和協(xié)議
NSObject不僅僅是一個類的名稱,還是一個協(xié)議的名稱。兩者對于定義一個Cocoa對象都是必要的。NSObject協(xié)議指定了Cocoa中所有根類必須的基本編程接口,因此不僅NSObject類采納了這個同名的協(xié)議,其它根類也采納這個協(xié)議,比如NSProxy。NSObject類進(jìn)一步指定了不作為代理對象的Cocoa對象的基本編程接口。
NSObject及類似的協(xié)議用于Cocoa對象的總體定義(而不是在類接口中包含那些協(xié)議),使多個根類成為可能。每個根類共用一個由它們采納的協(xié)議定義的公共接口。
在另一種意義上,NSObject不僅僅是個“根”協(xié)議。雖然NSObject類沒有正式采納NSCopying、NSMutableCopying、和NSCoding協(xié)議,但它聲明和實(shí)現(xiàn)了與那些協(xié)議相關(guān)的方法(而且,包含NSObject類的NSObject.h頭文件中也包含上面提到的所有四個協(xié)議的定義)。對象拷貝、編碼、和解碼是對象行為的基本部分。很多子類(如果不是絕大多數(shù)的話)都希望采納和遵循這些協(xié)議。
請注意:其它Cocoa類可以(而且確實(shí)是)通過范疇將方法添加到NSObject中。這些范疇通常是一些非正式的協(xié)議,在委托中使用。它們允許委托對象選擇實(shí)現(xiàn)范疇中的部分方法。然而,NSObject的范疇并不被認(rèn)為是基本對象接口的一部分。
根類方法概述
NSObject根類和它采納的NSObject協(xié)議及其它“根” 協(xié)議一起,為所有不作為代理對象的Cocoa對象指定了如下的接口和行為特征:
分配、初始化、和復(fù)制。NSObject類中的一些方法(包括一些來自協(xié)議的方法)用于對象的創(chuàng)建、初始化、和復(fù)制:
alloc和allocWithZone:方法用于從某內(nèi)存區(qū)域中分配一個對象內(nèi)存,并使對象指向其運(yùn)行時的類定義。
init方法是對象初始化原型,負(fù)責(zé)將對象的實(shí)例變量設(shè)置為一個已知的初始狀態(tài)。initialize和load是兩個類方法,它們讓對象有機(jī)會對自身進(jìn)行初始化。
new是一個將簡單的內(nèi)存分配和初始化結(jié)合起來的便利方法。
copy和copyWithZone:方法用于拷貝實(shí)現(xiàn)這些(由NSCopying協(xié)議定義的)方法的類的實(shí)例。希望支持可變對象拷貝的類則需要實(shí)現(xiàn)mutableCopy和mutableCopyWithZone:(由NSMutableCopying協(xié)議定義)方法。
更多信息請參見 "對象的創(chuàng)建"部分。
對象的保持和清理。下面的方法對面向?qū)ο蟪绦虻膬?nèi)存管理特別重要:
retain方法增加對象的保持次數(shù)。
release方法減少對象的保持次數(shù)。
autorelease方法也是減少對象的保持次數(shù),但是以推遲的方式。
retainCount方法返回對當(dāng)前的保持次數(shù)。
dealloc方法由需要釋放對象的實(shí)例變量以及釋放動態(tài)分配的內(nèi)存的類實(shí)現(xiàn)。
更多信息請參見 Cocoa對象的生命周期
內(nèi)省和比較。NSObjec有很多方法可以查詢對象的運(yùn)行時信息。這些內(nèi)省方法有助于找出對象在類層次中的位置,確定對象是否實(shí)現(xiàn)特定的方法,以及測試對象是否遵循某種協(xié)議。這些方法中的一部分僅實(shí)現(xiàn)為類方法。
superclass和class方法(實(shí)現(xiàn)為類和實(shí)例方法)分別以Class對象的形式返回接收者的超類和類。
您可以通過isKindOfClass:和isMemberOfClass:方法來確定對象屬于哪個類。后者用于測試接收者是否為指定類的實(shí)例。isSubclassOfClass:類方法則用于測試類的繼承性。
respondsToSelector:方法用于測試接收者是否實(shí)現(xiàn)由選擇器參數(shù)標(biāo)識的方法。instancesRespondToSelector:類方法則用于測試給定類的實(shí)例是否實(shí)現(xiàn)指定的方法。
conformsToProtocol:方法用于測試接收者(對象或類)是否遵循給定的協(xié)議。
isEqual:和hash方法用于對象的比較。
description方法允許對象返回一個內(nèi)容描述字符串;這個方法的輸出經(jīng)常用于調(diào)試(“print object”命令),以及在格式化字符串中和“%@”指示符一起表示對象。
更多信息請參見 內(nèi)省 。
對象的編碼和解碼。下面的方法和對象的編解碼(作為歸檔過程的一部分)有關(guān):
encodeWithCoder:和initWithCoder:是NSCoding協(xié)議僅有的方法。前者使對象可以對其實(shí)例變量進(jìn)行編碼,后者則使對象可以根據(jù)解碼過的實(shí)例變量對自身進(jìn)行初始化。
NSObject類中聲明了一些于對象編碼有關(guān)的方法:classForCoder:、replacementObjectForCoder:、和awakeAfterUsingCoder:。
進(jìn)一步信息請參見Cocoa的歸檔和序列化編程指南一文。
消息的轉(zhuǎn)發(fā)。forwardInvocation:和相關(guān)的方法允許一個對象將消息轉(zhuǎn)發(fā)給另一個對象。
消息的派發(fā)。以performSelector...開頭的一組方法使您可以在指定的延遲后派發(fā)消息,以及將消息從輔助線程派發(fā)(同步或異步)到主線程。
NSObject還有幾個其它的方法,包括一些處理版本和姿態(tài)(后者使一個類在運(yùn)行時將自己表示為另一個類)的類方法,以及一些訪問運(yùn)行時數(shù)據(jù)結(jié)構(gòu)的方法,比如方法選擇器和指向方法實(shí)現(xiàn)的函數(shù)指針。
接口規(guī)范
某些NSObject方法只是為了被調(diào)用,而另一些方法則是為了被重載。舉例來說,大多數(shù)子類不應(yīng)該重載allocWithZone:方法,但必須實(shí)現(xiàn)init方法—至少需要實(shí)現(xiàn)一個最終調(diào)用根類的init方法(請參見"對象的創(chuàng)建"部分)的初始化方法。對于那些期望子類重載的方法,NSObject的實(shí)現(xiàn)或者什么也不做,或者返回一個合理的值,比如self。這些缺省實(shí)現(xiàn)使我們有可能向任意的Cocoa對象—甚至是沒有重載這些方法的對象—發(fā)送諸如init這樣得基本消息,而又不必冒運(yùn)行時例外的風(fēng)險。在發(fā)送消息之前,不必進(jìn)行檢查(通過respondsToSelector:方法)。更加重要的是,NSObject的這些“占位”方法為Cocoa對象定義了一個公共的結(jié)構(gòu),并建立了一些規(guī)則,如果所有的對象都遵循這些規(guī)則,對象間的交互將更加可靠。
實(shí)例方法和類方法
運(yùn)行環(huán)境系統(tǒng)以一種特殊的方式處理根類定義的方法。根類定義的實(shí)例方法可以由實(shí)例對象和類對象執(zhí)行,因此所有類對象都可以訪問根類定義的實(shí)例方法。對于任何類對象,如果對象中不包含同名的類方法,就可以執(zhí)行根類的所有實(shí)例方法。
舉例來說,一個類對象可以通過發(fā)送消息來執(zhí)行NSObject的respondsToSelector:和performSelector:withObject:實(shí)例方法:
- SEL method = @selector(riskAll:);
- if ([MyClass respondsToSelector:method])
- [MyClass performSelector:method withObject:self];
請注意,只有根類中定義的實(shí)例方法才可以在類對象中使用。在上面的例子中,如果MyClass重新實(shí)現(xiàn)了respondsToSelector:或者performSelector:withObject:方法,則那些新的版本將只能用于實(shí)例對象。MyClass的類對象只能執(zhí)行NSObject類定義的版本(當(dāng)然,如果MyClass將respondsToSelector:或performSelector:withObject: 實(shí)現(xiàn)為類方法,而不是實(shí)例方法,則該類對象可以執(zhí)行這些新的實(shí)現(xiàn))。
小結(jié):Cocoa對象 根類的內(nèi)容介紹完了,希望本文對你 有所幫助!