首先感謝這幾個(gè)篇文章對(duì)我的幫助:
http://blog.csdn.net/mangosnow/article/details/36183535
http://blog.sina.com.cn/s/blog_71e456db0100w1bm.html
http://book.51cto.com/art/201403/432146.htm
http://www.itqx.net/thread-2286-1-1.html
http://blog.csdn.net/c395565746c/article/details/8507008
上面幾篇文章都是在網(wǎng)上查閱到的資料
接下來(lái)乘寒,我們要通過(guò)一個(gè)小例子來(lái)簡(jiǎn)單错沃、通俗的理解一下什么是消息轉(zhuǎn)發(fā)以及如何消息轉(zhuǎn)發(fā),希望看完這篇文章時(shí)大家會(huì)徹底的明白OC的消息纵揍。
首先,你需要知道這兩個(gè)概念:
OC中調(diào)用方法就是向?qū)ο蟀l(fā)送消息嗅剖。
比如 :
[person run];
這實(shí)際上這是在給person這個(gè)對(duì)象發(fā)送run這個(gè)消息范删。
那么問(wèn)題來(lái)了,當(dāng)run這個(gè)方法只有定義沒(méi)有實(shí)現(xiàn)會(huì)怎么樣呢脓杉?
就是經(jīng)典的報(bào)錯(cuò)
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Person run]: unrecognized selector sent to instance
ok糟秘,前提已經(jīng)說(shuō)完了,我們就從找這個(gè)錯(cuò)誤原因講起球散。
首先尿赚,該方法在調(diào)用時(shí),系統(tǒng)會(huì)查看這個(gè)對(duì)象能否接收這個(gè)消息(查看這個(gè)類(lèi)有沒(méi)有這個(gè)方法蕉堰,或者有沒(méi)有實(shí)現(xiàn)這個(gè)方法凌净。),如果不能并且只在不能的情況下屋讶,就會(huì)調(diào)用下面這幾個(gè)方法冰寻,給你“補(bǔ)救”的機(jī)會(huì),你可以先理解為幾套防止程序crash的備選方案皿渗,我們就是利用這幾個(gè)方案進(jìn)行消息轉(zhuǎn)發(fā)斩芭,注意一點(diǎn),前一套方案實(shí)現(xiàn)后一套方法就不會(huì)執(zhí)行乐疆。如果這幾套方案你都沒(méi)有做處理划乖,那么程序就會(huì)報(bào)錯(cuò)crash。
打個(gè)比方:比賽足球時(shí)挤土,腳下有球的那名球員琴庵,如果他的位置不利于射門(mén)或者他的球即將被對(duì)方球員搶斷,這時(shí)最好是把球傳出去仰美,這里的球就相當(dāng)于消息迷殿。
方案一:
- + (BOOL)resolveInstanceMethod:(SEL)sel
- + (BOOL)resolveClassMethod:(SEL)sel
方案二:
- - (id)forwardingTargetForSelector:(SEL)aSelector
方案三:
- - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;
- - (void)forwardInvocation:(NSInvocation *)anInvocation;
到目前為止大家已經(jīng)知道什么是消息轉(zhuǎn)發(fā)了。下面就說(shuō)一下這幾套方案是怎樣調(diào)用的咖杂。
首先庆寺,系統(tǒng)會(huì)調(diào)用resolveInstanceMethod(當(dāng)然,如果這個(gè)方法是一個(gè)類(lèi)方法翰苫,就會(huì)調(diào)用resolveClassMethod)讓你自己為這個(gè)方法增加實(shí)現(xiàn)止邮。
咱們來(lái)看一個(gè)例子:
首先这橙,創(chuàng)建了一個(gè)Person類(lèi)的對(duì)象p,然后調(diào)用p的run方法导披,注意屈扎,這個(gè)run方法是沒(méi)有寫(xiě)實(shí)現(xiàn)的。
進(jìn)入Person類(lèi)的.m文件撩匕,我實(shí)現(xiàn)了resolveInstanceMethod這個(gè)方法為我的Person類(lèi)動(dòng)態(tài)增加了一個(gè)run方法的實(shí)現(xiàn)鹰晨。(什么是動(dòng)態(tài)增加?其實(shí)就是在程序運(yùn)行的時(shí)候給某類(lèi)的某個(gè)方法增加實(shí)現(xiàn)止毕。具體實(shí)現(xiàn)內(nèi)容就為上面的void run 這個(gè)c函數(shù)模蜡。)
當(dāng)外部調(diào)用[p run]時(shí),由于我們沒(méi)有實(shí)現(xiàn)run對(duì)應(yīng)的方法扁凛,那么系統(tǒng)會(huì)調(diào)用resolveInstanceMethod讓你去做一些其他操作忍疾。(當(dāng)然,你也可以不做操作谨朝,只是在這個(gè)例子中卤妒,我為run方法動(dòng)態(tài)增加了實(shí)現(xiàn)。)
繼續(xù)運(yùn)行字币,程序走到了我們C函數(shù)的部分则披,這樣程序沒(méi)有了崩潰。
下面講一下第二套方法洗出,forwardingTargetForSelector士复,這個(gè)方法返回你需要轉(zhuǎn)發(fā)消息的對(duì)象。
我們接著這個(gè)例子來(lái)講翩活,為了便于演示消息轉(zhuǎn)發(fā)阱洪,我們新建了一個(gè)汽車(chē)類(lèi)Car,并且實(shí)現(xiàn)了Car的run方法隅茎。
現(xiàn)在我不去對(duì)方案一的resolveInstanceMethod做任何處理澄峰,直接調(diào)用父類(lèi)方法嫉沽”傧可以看到,系統(tǒng)已經(jīng)來(lái)到了forwardingTargetForSelector方法绸硕,我們現(xiàn)在返回一個(gè)Car類(lèi)的實(shí)例對(duì)象堂竟。
繼續(xù)運(yùn)行,程序就來(lái)到了Car類(lèi)的run方法玻佩,這樣出嘹,我們就實(shí)現(xiàn)了消息轉(zhuǎn)發(fā)。
繼續(xù)我們的例子咬崔。如果我們不實(shí)現(xiàn)forwardingTargetForSelector税稼,系統(tǒng)就會(huì)調(diào)用方案三的兩個(gè)方法methodSignatureForSelector和forwardInvocation
methodSignatureForSelector用來(lái)生成方法簽名烦秩,這個(gè)簽名就是給forwardInvocation中的參數(shù)NSInvocation調(diào)用的。
開(kāi)頭我們要找的錯(cuò)誤unrecognized selector sent to instance原因郎仆,原來(lái)就是因?yàn)閙ethodSignatureForSelector這個(gè)方法中只祠,由于沒(méi)有找到run對(duì)應(yīng)的實(shí)現(xiàn)方法,所以返回了一個(gè)空的方法簽名扰肌,最終導(dǎo)致程序報(bào)錯(cuò)崩潰抛寝。
所以我們需要做的是自己新建方法簽名,再在forwardInvocation中用你要轉(zhuǎn)發(fā)的那個(gè)對(duì)象調(diào)用這個(gè)對(duì)應(yīng)的簽名曙旭,這樣也實(shí)現(xiàn)了消息轉(zhuǎn)發(fā)盗舰。
關(guān)于生成簽名的類(lèi)型"v@:"解釋一下。每一個(gè)方法會(huì)默認(rèn)隱藏兩個(gè)參數(shù)桂躏,self钻趋、_cmd,self代表方法調(diào)用者剂习,_cmd代表這個(gè)方法的SEL爷绘,簽名類(lèi)型就是用來(lái)描述這個(gè)方法的返回值、參數(shù)的进倍,v代表返回值為void土至,@表示self,:表示_cmd猾昆。
現(xiàn)在我們回到最初陶因,我們調(diào)用的是Person類(lèi)的run方法,最終方法被Car類(lèi)的對(duì)象來(lái)接受垂蜗。這就是OC的消息轉(zhuǎn)發(fā)機(jī)制楷扬。