第十二條:理解消息轉(zhuǎn)發(fā)機(jī)制
上一條我們說(shuō)了對(duì)象的消息傳遞機(jī)制很重要,那么問(wèn)題來(lái)了----對(duì)象在收到無(wú)法解讀的消息之后會(huì)發(fā)生什么呢?
若想令類能理解某條消息脸狸,我們必須以程序碼實(shí)現(xiàn)出對(duì)應(yīng)的方法才行们镜。
但是,在編譯器向類發(fā)送了其無(wú)法解讀的消息并不會(huì)報(bào)錯(cuò)矢洲,因?yàn)樵谶\(yùn)行期可以繼續(xù)向類中添加方法,所以編譯器在編譯時(shí)還無(wú)法確知類中到底會(huì)不會(huì)有某個(gè)方法實(shí)現(xiàn)缩焦。
當(dāng)對(duì)象接收到無(wú)法解讀的消息后读虏,就會(huì)啟動(dòng)“消息轉(zhuǎn)發(fā)”(message forwarding)機(jī)制责静,程序員可經(jīng)由此過(guò)程告訴對(duì)象應(yīng)該如何處理未知消息。
消息轉(zhuǎn)發(fā)過(guò)程分為兩大階段:
第一階段先征詢接收者盖桥,所屬的類灾螃,看其是否能動(dòng)態(tài)添加方法,以處理當(dāng)前這個(gè)“未知的選擇子”(unknown selector)揩徊,這叫做“動(dòng)態(tài)方法解析”(dynamic method resolution)腰鬼。
第二階段涉及“完整的消息轉(zhuǎn)發(fā)機(jī)制”(full forwarding mechanism)。如果運(yùn)行期系統(tǒng)已經(jīng)把第一階段執(zhí)行完了塑荒,那么接收者自己就無(wú)法再以動(dòng)態(tài)新增方法的手段來(lái)響應(yīng)包含該選擇子的消息了熄赡。
此時(shí),運(yùn)行時(shí)系統(tǒng)會(huì)請(qǐng)求接收者以其他手段來(lái)處理與消息相關(guān)的方法調(diào)用齿税。
這又分為兩步:首先彼硫,請(qǐng)接收者看看有沒(méi)有其他對(duì)象能處理這條消息。
若有凌箕,則運(yùn)行時(shí)系統(tǒng)會(huì)把消息轉(zhuǎn)給那個(gè)對(duì)象拧篮,于是消息轉(zhuǎn)發(fā)過(guò)程解析,一切正常牵舱。
若沒(méi)有“備援的接受者”(replacement receiver)串绩,則啟動(dòng)完整的消息轉(zhuǎn)發(fā)機(jī)制,運(yùn)行時(shí)系統(tǒng)會(huì)把與消息有關(guān)的全部細(xì)節(jié)都封裝到NSInvocation對(duì)象中芜壁,再給接收者最后一次機(jī)會(huì)赏参,令其設(shè)法解決當(dāng)前還未處理的這條消息。
動(dòng)態(tài)方法解析
對(duì)象在收到無(wú)法解讀的消息后沿盅,首先將調(diào)用其所述類的下列類方法:
+(BOOL)resolveInstanceMethod:(SEL)selector;
該方法的參數(shù)就是那個(gè)未知的選擇子把篓,其返回值為Boolean類型,表示這個(gè)類是否能新增一個(gè)實(shí)例方法用以處理此選擇子腰涧。
再繼續(xù)往下執(zhí)行轉(zhuǎn)發(fā)機(jī)制之前韧掩,本類有機(jī)會(huì)新增一個(gè)處理此選擇子的方法。
假如尚未實(shí)現(xiàn)的方法不是實(shí)例方法而是類方法窖铡,那么運(yùn)行期系統(tǒng)就會(huì)調(diào)用另外一個(gè)方法疗锐,該方法與“resolveInstanceMethod:”類似,叫做“resolveClassMethod”费彼。
使用這種辦法的前提是:相關(guān)方法的實(shí)現(xiàn)代碼已經(jīng)寫(xiě)好滑臊,只等著運(yùn)行的時(shí)候動(dòng)態(tài)插在類里面就可以了。
此方案常用來(lái)實(shí)現(xiàn)@dynamic屬性
要點(diǎn):
1.若對(duì)象無(wú)法響應(yīng)某個(gè)選擇子箍铲,則進(jìn)入消息轉(zhuǎn)發(fā)流程雇卷。
2.通過(guò)運(yùn)行期的動(dòng)態(tài)方法解析功能,我們可以在需要用到某個(gè)方法時(shí)再將其加入類中。
3.對(duì)象可以把其無(wú)法解讀的某些選擇子轉(zhuǎn)交給其他對(duì)象來(lái)處理关划。
4.講過(guò)上述兩步之后小染,如果還沒(méi)辦法處理選擇子,那就啟動(dòng)完整的消息轉(zhuǎn)發(fā)機(jī)制贮折。