Objective-C 的“多繼承”

當(dāng)單繼承不夠用,很難為問題域建模時(shí)馋艺,我們通常都會(huì)直接想到多繼承嚎卫。多繼承是從多余一個(gè)直接基類派生類的能力嘉栓,可以更加直接地為應(yīng)用程序建模。但是Objective-C不支持多繼承拓诸,由于消息機(jī)制名字查找發(fā)生在運(yùn)行時(shí)而非編譯時(shí)侵佃,很難解決多個(gè)基類可能導(dǎo)致的二義性問題。不過其實(shí) Objective-C 也無需支持多繼承奠支,我們可以找到如下幾種間接實(shí)現(xiàn)多繼承目的的方法:

消息轉(zhuǎn)發(fā)??

delegate和protocol ?

類別? ?

當(dāng)向someObject發(fā)送某消息馋辈,但runtime system在當(dāng)前類和父類中都找不到對應(yīng)方法的實(shí)現(xiàn)時(shí),runtime system并不會(huì)立即報(bào)錯(cuò)使程序崩潰倍谜,而是依次執(zhí)行下列步驟:

分別簡述一下流程:

1.動(dòng)態(tài)方法解析:向當(dāng)前類發(fā)送 resolveInstanceMethod: 信號首有,檢查是否動(dòng)態(tài)向該類添加了方法燕垃。(迷茫請搜索:@dynamic)

2.快速消息轉(zhuǎn)發(fā):檢查該類是否實(shí)現(xiàn)了 forwardingTargetForSelector: 方法,若實(shí)現(xiàn)了則調(diào)用這個(gè)方法井联。若該方法返回值對象非nil或非self卜壕,則向該返回對象重新發(fā)送消息。

3.標(biāo)準(zhǔn)消息轉(zhuǎn)發(fā):runtime發(fā)送methodSignatureForSelector:消息獲取Selector對應(yīng)的方法簽名烙常。返回值非空則通過forwardInvocation:轉(zhuǎn)發(fā)消息轴捎,返回值為空則向當(dāng)前對象發(fā)送doesNotRecognizeSelector:消息,程序崩潰退出蚕脏。

顧名思義侦副,我們可以利用上述過程中的2、3兩種方式來完成消息轉(zhuǎn)發(fā)驼鞭。

快速消息轉(zhuǎn)發(fā)

快速消息轉(zhuǎn)發(fā)的實(shí)現(xiàn)方法很簡單秦驯,只需要重寫 - (id)forwardingTargetForSelector:(SEL)aSelector ?方法即可。

我來舉個(gè)簡單的例子挣棕,比如現(xiàn)有2個(gè)類:Teacher 和 Doctor译隘,Doctor可以做手術(shù)(operate方法)。

@interface?Teacher?:?NSObject? ?

@end? ?

@interface?Doctor?:?NSObject? ?

-?(void)operate;???

@end???

通過快速消息轉(zhuǎn)發(fā)洛心,可以很輕松的讓teacher調(diào)用doctor的方法做手術(shù)固耘。

Teacher類需要實(shí)現(xiàn)將消息轉(zhuǎn)發(fā)給Doctor:

-?(id)forwardingTargetForSelector:(SEL)aSelector???

{???

????Doctor?*doctor?=?[[Doctor?alloc]init];???

if?([doctor?respondsToSelector:aSelector])?{???

return?doctor;???

????}???

return?nil;???

}???

雖然消息可以動(dòng)態(tài)轉(zhuǎn)發(fā)傳遞,但是編輯器的靜態(tài)檢查是繞不過的词身,那么問題來了厅目,既然Teacher類沒有實(shí)現(xiàn)operate方法又該如何聲明呢?

到目前為止法严,我只想到下面2種方法:

聲明方法1 ———— 類別

@interface?Teacher?(DoctorMethod)???

-?(void)operate;? ?

@end? ?

聲明方法2 ———— 導(dǎo)入頭文件损敷、調(diào)用時(shí)強(qiáng)轉(zhuǎn)類型

Teacher類頭文件需要包含Doctor頭文件,告訴編譯器去Doctor.h中可以找到operator方法的聲明深啤,并且在調(diào)用時(shí)強(qiáng)轉(zhuǎn)類型嗤锉。

Teacher?*teacher?=?[[Teacher?alloc]init];???

[(Doctor?*)teacher?operate];???

有興趣可以思考一個(gè)問題:如果將其類型轉(zhuǎn)成 id ,也可以編譯通過墓塌,并實(shí)現(xiàn)轉(zhuǎn)發(fā)“露睿可是會(huì)帶來什么隱患呢苫幢?

方法1使用類別足夠清晰簡便,為什么還要提出辦法2呢 垫挨? 我的想法是韩肝,方法1的弊端是拋出來的方法是定死的,而且在.h里露著九榔;方法2就相對靈活哀峻,而且隱藏了我要轉(zhuǎn)發(fā)的消息涡相。

標(biāo)準(zhǔn)消息轉(zhuǎn)發(fā)

標(biāo)準(zhǔn)消息轉(zhuǎn)發(fā)需要重寫 methodSignatureForSelector: 和 forwardInvocation: 兩個(gè)方法即可。

轉(zhuǎn)發(fā)重寫方法:

-?(NSMethodSignature?*)methodSignatureForSelector:(SEL)aSelector???

{???

????NSMethodSignature*?signature?=?[super?methodSignatureForSelector:aSelector];???

if?(signature==nil)?{???

????????signature?=?[someObj?methodSignatureForSelector:aSelector];???

????}???

????NSUInteger?argCount?=?[signature?numberOfArguments];???

for(NSInteger?i=0?;?i

????}? ?

return?signature;???

}? ?

-?(void)forwardInvocation:(NSInvocation?*)anInvocation???

{???

????SEL?seletor?=?[anInvocation?selector];???

if?([someObj?respondsToSelector:seletor])?{???

????????[anInvocation?invokeWithTarget:someObj];???

????}? ?

}???

兩種消息轉(zhuǎn)發(fā)方式的比較

快速消息轉(zhuǎn)發(fā):簡單剩蟀、快速催蝗、但僅能轉(zhuǎn)發(fā)給一個(gè)對象。

標(biāo)準(zhǔn)消息轉(zhuǎn)發(fā):稍復(fù)雜育特、較慢丙号、但轉(zhuǎn)發(fā)操作實(shí)現(xiàn)可控,可以實(shí)現(xiàn)多對象轉(zhuǎn)發(fā)缰冤。

delegate和protocol

委托是Objective-C中最常用的一種回調(diào)機(jī)制犬缨。用法我覺得沒什么好說的,總結(jié)一下該機(jī)制特點(diǎn):

委托協(xié)助主體完成操作任務(wù)棉浸,將需要定制化的操作預(yù)留給委托對象來自定義實(shí)現(xiàn)怀薛,類似子類化主體。

除此之外迷郑,可以用作事件監(jiān)聽枝恋。

一時(shí)還真想不出來了…

類別

個(gè)人認(rèn)為類別是Objective-C設(shè)計(jì)的一大精髓,也是我愛上Objective-C的最大理由三热。

類別是個(gè)強(qiáng)大的東西鼓择,它既可以為類添加方法,也可以添加實(shí)例就漾。一定有不少人不認(rèn)同呐能,想提醒我:類別的局限性之一就是無法向類中添加新的實(shí)例變量。 背書真心毀人抑堡,聽我舉個(gè)例子慢慢說摆出。

重新再來個(gè)Teacher類:

@interface?Teacher?:?NSObject???

{???

????NSUInteger?age;???

}? ?

@end???

光有個(gè)年齡還不能滿足對teacher的描述,我想加個(gè)profession實(shí)例來存teacher的專業(yè)首妖。直觀的想法是子類化Teacher偎漫,其實(shí)也可以用類別。

你需要了解一下 runtime 編程知識有缆,關(guān)注一下 objc_setAssociatedObject 和 objc_getAssociatedObject 象踊。

//??Teacher+Profession.m? ?

#import?"Teacher+Profession.h"???

#import? ?

const?char?*ProfessionType?=?"NSString?*";???

@implementation?Teacher?(Profession)? ?

-(void)setProf:(NSString*)prof???

{???

????objc_setAssociatedObject(self,?ProfessionType,?prof,?OBJC_ASSOCIATION_RETAIN_NONATOMIC);???

}? ?

-(NSString?*)prof???

{???

????NSString?*pro?=?objc_getAssociatedObject(self,?ProfessionType);???

return?pro;???

}? ?

@end???

現(xiàn)在就可以通過setProf: 和 prof 來存取 teacher 的 profession 值了。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末棚壁,一起剝皮案震驚了整個(gè)濱河市杯矩,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌袖外,老刑警劉巖史隆,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異曼验,居然都是意外死亡泌射,警方通過查閱死者的電腦和手機(jī)粘姜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來熔酷,“玉大人孤紧,你說我怎么就攤上這事〈吭桑” “怎么了坛芽?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長翼抠。 經(jīng)常有香客問我咙轩,道長,這世上最難降的妖魔是什么阴颖? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任活喊,我火速辦了婚禮,結(jié)果婚禮上量愧,老公的妹妹穿的比我還像新娘钾菊。我一直安慰自己,他們只是感情好偎肃,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布煞烫。 她就那樣靜靜地躺著,像睡著了一般累颂。 火紅的嫁衣襯著肌膚如雪滞详。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天紊馏,我揣著相機(jī)與錄音料饥,去河邊找鬼。 笑死朱监,一個(gè)胖子當(dāng)著我的面吹牛岸啡,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播赫编,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼巡蘸,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了擂送?” 一聲冷哼從身側(cè)響起悦荒,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎团甲,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體黍聂,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡躺苦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年身腻,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片匹厘。...
    茶點(diǎn)故事閱讀 38,137評論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡嘀趟,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出愈诚,到底是詐尸還是另有隱情她按,我是刑警寧澤,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布炕柔,位于F島的核電站酌泰,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏匕累。R本人自食惡果不足惜陵刹,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望欢嘿。 院中可真熱鬧衰琐,春花似錦、人聲如沸炼蹦。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽掐隐。三九已至狗热,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間瑟枫,已是汗流浹背斗搞。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留慷妙,地道東北人僻焚。 一個(gè)月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像膝擂,于是被迫代替她去往敵國和親虑啤。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評論 2 345