當單繼承不夠用,很難為問題域建模時,我們通常都會直接想到多繼承。多繼承是從多餘一個直接基類派生類的能力,可以更加直接地為應用程式建模。但是Objective-C不支援多繼承,由於訊息機制名字查詢發生在執行時而非編譯時,很難解決多個基類可能導致的二義性問題。不過其實 Objective-C 也無需支援多繼承,我們可以找到如下幾種間接實現多繼承目的的方法:
delegate和protocol
類別
當向someObject傳送某訊息,但runtime system在當前類和父類中都找不到對應方法的實現時,runtime system並不會立即報錯使程式崩潰,而是依次執行下列步驟:
分別簡述一下流程:
1.動態方法解析
向當前類傳送 resolveInstanceMethod: 訊號,檢查是否動態向該類添加了方法。(迷茫請搜尋:@dynamic)
檢查該類是否實現了 forwardingTargetForSelector: 方法,若實現了則呼叫這個方法。若該方法返回值物件非nil或非self,則向該返回物件重新發送訊息。
runtime傳送methodSignatureForSelector:訊息獲取Selector對應的方法簽名。返回值非空則透過forwardInvocation:轉發訊息,返回值為空則向當前物件傳送doesNotRecognizeSelector:訊息,程式崩潰退出。
我來舉個簡單的例子,比如現有2個類:Teacher 和 Doctor,Doctor可以做手術(operate方法)。
@interface Teacher : NSObject
@end
@interface Doctor : NSObject
- (void)operate;
- (id)forwardingTargetForSelector:(SEL)aSelector
{
Doctor *doctor = [[Doctor alloc]init];
if ([doctor respondsToSelector:aSelector]) {
return doctor;
}
return nil;
到目前為止,我只想到下面2種方法:
宣告方法1 ———— 類別
@interface Teacher (DoctorMethod)
宣告方法2 ———— 匯入標頭檔案、呼叫時強轉型別
Teacher類標頭檔案需要包含Doctor標頭檔案,告訴編譯器去Doctor.h中可以找到operator方法的宣告,並且在呼叫時強轉型別。
Teacher *teacher = [[Teacher alloc]init];
[(Doctor *)teacher operate];
當單繼承不夠用,很難為問題域建模時,我們通常都會直接想到多繼承。多繼承是從多餘一個直接基類派生類的能力,可以更加直接地為應用程式建模。但是Objective-C不支援多繼承,由於訊息機制名字查詢發生在執行時而非編譯時,很難解決多個基類可能導致的二義性問題。不過其實 Objective-C 也無需支援多繼承,我們可以找到如下幾種間接實現多繼承目的的方法:
delegate和protocol
類別
當向someObject傳送某訊息,但runtime system在當前類和父類中都找不到對應方法的實現時,runtime system並不會立即報錯使程式崩潰,而是依次執行下列步驟:
分別簡述一下流程:
1.動態方法解析
向當前類傳送 resolveInstanceMethod: 訊號,檢查是否動態向該類添加了方法。(迷茫請搜尋:@dynamic)
檢查該類是否實現了 forwardingTargetForSelector: 方法,若實現了則呼叫這個方法。若該方法返回值物件非nil或非self,則向該返回物件重新發送訊息。
runtime傳送methodSignatureForSelector:訊息獲取Selector對應的方法簽名。返回值非空則透過forwardInvocation:轉發訊息,返回值為空則向當前物件傳送doesNotRecognizeSelector:訊息,程式崩潰退出。
我來舉個簡單的例子,比如現有2個類:Teacher 和 Doctor,Doctor可以做手術(operate方法)。
@interface Teacher : NSObject
@end
@interface Doctor : NSObject
- (void)operate;
@end
- (id)forwardingTargetForSelector:(SEL)aSelector
{
Doctor *doctor = [[Doctor alloc]init];
if ([doctor respondsToSelector:aSelector]) {
return doctor;
}
return nil;
}
到目前為止,我只想到下面2種方法:
宣告方法1 ———— 類別
@interface Teacher (DoctorMethod)
- (void)operate;
@end
宣告方法2 ———— 匯入標頭檔案、呼叫時強轉型別
Teacher類標頭檔案需要包含Doctor標頭檔案,告訴編譯器去Doctor.h中可以找到operator方法的宣告,並且在呼叫時強轉型別。
Teacher *teacher = [[Teacher alloc]init];
[(Doctor *)teacher operate];