輕鬆學習之一--Objective-C訊息轉發
文/J_雨(簡書作者)
原文連結:http://www.jianshu.com/p/1bde36ad9938
首先感謝這幾個篇文章對我的幫助:
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
上面幾篇文章都是在網上查閱到的資料
接下來,我們要通過一個小例子來簡單、通俗的理解一下什麼是訊息轉發以及如何訊息轉發,希望看完這篇文章時大家會徹底的明白OC的訊息。
首先,你需要知道這兩個概念:
OC中呼叫方法就是向物件傳送訊息。
比如 :
[person run];
這實際上這是在給person這個物件傳送run這個訊息。
那麼問題來了,當run這個方法只有定義沒有實現會怎麼樣呢?
就是經典的報錯
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Person run]: unrecognized selector sent to instance
ok,前提已經說完了,我們就從找這個錯誤原因講起。
首先,該方法在呼叫時,系統會檢視這個物件能否接收這個訊息(檢視這個類有沒有這個方法,或者有沒有實現這個方法。),如果不能並且只在不能的情況下,就會呼叫下面這幾個方法,給你“補救”的機會,你可以先理解為幾套防止程式crash的備選方案,我們就是利用這幾個方案進行訊息轉發,注意一點,前一套方案實現後一套方法就不會執行。如果這幾套方案你都沒有做處理,那麼程式就會報錯crash。
打個比方:比賽足球時,腳下有球的那名球員,如果他的位置不利於射門或者他的球即將被對方球員搶斷,這時最好是把球傳出去,這裡的球就相當於訊息。
方案一:
- + (BOOL)resolveInstanceMethod:(SEL)sel
- + (BOOL)resolveClassMethod:(SEL)sel
方案二:
- - (id)forwardingTargetForSelector:(SEL)aSelector
方案三:
- - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;
- - (void)forwardInvocation:(NSInvocation *)anInvocation;
到目前為止大家已經知道什麼是訊息轉發了。下面就說一下這幾套方案是怎樣呼叫的。
首先,系統會呼叫resolveInstanceMethod(當然,如果這個方法是一個類方法,就會呼叫resolveClassMethod)讓你自己為這個方法增加實現。
我們們來看一個例子:
首先,建立了一個Person類的物件p,然後呼叫p的run方法,注意,這個run方法是沒有寫實現的。
進入Person類的.m檔案,我實現了resolveInstanceMethod這個方法為我的Person類動態增加了一個run方法的實現。(什麼是動態增加?其實就是在程式執行的時候給某類的某個方法增加實現。具體實現內容就為上面的void run 這個c函式。)
當外部呼叫[p run]時,由於我們沒有實現run對應的方法,那麼系統會呼叫resolveInstanceMethod讓你去做一些其他操作。(當然,你也可以不做操作,只是在這個例子中,我為run方法動態增加了實現。)
繼續執行,程式走到了我們C函式的部分,這樣程式沒有了崩潰。
下面講一下第二套方法,forwardingTargetForSelector,這個方法返回你需要轉發訊息的物件。
我們接著這個例子來講,為了便於演示訊息轉發,我們新建了一個汽車類Car,並且實現了Car的run方法。
現在我不去對方案一的resolveInstanceMethod做任何處理,直接呼叫父類方法。可以看到,系統已經來到了forwardingTargetForSelector方法,我們現在返回一個Car類的例項物件。
繼續執行,程式就來到了Car類的run方法,這樣,我們就實現了訊息轉發。
繼續我們的例子。如果我們不實現forwardingTargetForSelector,系統就會呼叫方案三的兩個方法methodSignatureForSelector和forwardInvocation
methodSignatureForSelector用來生成方法簽名,這個簽名就是給forwardInvocation中的引數NSInvocation呼叫的。
開頭我們要找的錯誤unrecognized selector sent to instance原因,原來就是因為methodSignatureForSelector這個方法中,由於沒有找到run對應的實現方法,所以返回了一個空的方法簽名,最終導致程式報錯崩潰。
所以我們需要做的是自己新建方法簽名,再在forwardInvocation中用你要轉發的那個物件呼叫這個對應的簽名,這樣也實現了訊息轉發。
關於生成簽名的型別"v@:"解釋一下。每一個方法會預設隱藏兩個引數,self、_cmd,self代表方法呼叫者,_cmd代表這個方法的SEL,簽名型別就是用來描述這個方法的返回值、引數的,v代表返回值為void,@表示self,:表示_cmd。
現在我們回到最初,我們呼叫的是Person類的run方法,最終方法被Car類的物件來接受。這就是OC的訊息轉發機制。
相關文章
- XML輕鬆學習手冊(3)XML概念之一(轉)XML
- Objective-C中的訊息轉發Object
- 智慧運營助你輕鬆玩轉訊息推送
- Objective-C Runtime (二):方法與訊息轉發Object
- 輕鬆上手 PHP + RabbitMQ 訊息釋出與訂閱PHPMQ
- 怎麼輕鬆學習JavaScriptJavaScript
- 專案管理輕鬆學(轉)專案管理
- iOS-訊息轉發和方法調配技術學習iOS
- 如何輕鬆學習 Kubernetes?
- XML輕鬆學習手冊(6)XML例項解析(轉)XML
- 乾貨|輕鬆學DDD之一:模型驅動設計模型
- iOS 訊息轉發iOS
- XML輕鬆學習手冊(5)XML語法之二(轉)XML
- XML輕鬆學習手冊(5)XML語法之四(轉)XML
- Yii2 - Active Record 輕鬆學習
- OC訊息機制,訊息轉發機制
- Python輕鬆檢視微信撤回訊息,祕密無處可藏Python
- 3個步驟輕鬆整合Push Kit,實現App訊息推送APP
- 輕鬆開發多語言程式 (轉)
- 辦公室革命,教你輕鬆搞定輕鬆玩轉ExcelExcel
- XML輕鬆學習手冊(4)XML的術語之二(轉)XML
- XML輕鬆學習手冊(2)XML快速入門之二(轉)XML
- XML輕鬆學習手冊(2)XML快速入門之三(轉)XML
- redis學習(七) 訊息通知Redis
- 如何輕鬆學習Python資料分析?Python
- 萬字長文,帶你輕鬆學習 SparkSpark
- 輕鬆學習 JavaScript(8):JavaScript 中的類JavaScript
- iOS訊息轉發小記iOS
- 輕鬆學 dva(基礎 + 進階)(32 個視訊)
- AI 學習之路——輕鬆初探 Python 篇(三)AIPython
- 輕鬆學習 JavaScript(1):瞭解 let 語句JavaScript
- 輕鬆學習 JavaScript(6):JavaScript 箭頭函式JavaScript函式
- 輕鬆學習 JavaScript(5):簡化函式提升JavaScript函式
- AI 學習之路——輕鬆初探 Python 篇(一)AIPython
- 女神把微信訊息撤回後好慌,Python幾十行程式碼輕鬆檢視撤回訊息!Python行程
- 訊息佇列學習腦圖佇列
- 訊息佇列學習基礎佇列
- MFC學習(四) 訊息機制