老規(guī)矩饶唤,先上經(jīng)典圖
消息在繼承樹上沒有找到實現(xiàn)還有消息轉(zhuǎn)發(fā)提供三次機會歉糜,不至于直接報doesNotRecognizeSelector錯誤崩潰,當(dāng)然這也給線上熱修復(fù)提供了支持弊攘。
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
[self performSelector:@selector(drink)];
}
繼承樹上沒有drink方法八堡,則進(jìn)入消息轉(zhuǎn)發(fā)階段
一桥温、動態(tài)方法解析 resolveInstanceMethod
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
if ([NSStringFromSelector(sel) isEqualToString:@"drink"])
{
NSLog(@"drink 動態(tài)方法解析");
class_addMethod([self class], sel, (IMP)drinkMethod, nil);
return YES;
}
return [super resolveInstanceMethod:sel];
}
void drinkMethod()
{
NSLog(@"動態(tài)方法添加 drink");
}
通過runtime動態(tài)添加drinkMethod方法來實現(xiàn)扑浸,這個類方法只要動態(tài)增加了方法烧给,不管return YES或者NO都不會繼續(xù)往下走。
二喝噪、備用接受者
如果resolveInstanceMethod沒有動態(tài)添加方法就會繼續(xù)往下走forwardingTargetForSelector础嫡,將消息轉(zhuǎn)發(fā)給其他對象來實現(xiàn)。
- (id)forwardingTargetForSelector:(SEL)aSelector
{
if ([NSStringFromSelector(aSelector) isEqualToString:@"drink"])
{
NSLog(@"備用接收者");
return [Son new];
}
return [super forwardingTargetForSelector:aSelector];
}
如果Son類中有drink方法酝惧,則會執(zhí)行榴鼎,如果沒有drink則會報doesNotRecognizeSelector崩潰。如果forwardingTargetForSelector方法返回nil或者沒有實現(xiàn)晚唇,則會進(jìn)入最后一步完整消息轉(zhuǎn)發(fā)巫财。
3、完整消息轉(zhuǎn)發(fā)
如果前兩步都未處理哩陕,則會進(jìn)入最后的機會平项,先調(diào)用methodSignatureForSelector獲取方法簽名,每個字符都代表一種參數(shù)萌踱,具體對應(yīng)表格可參考官方文檔方法簽名,也可以直接使用類型編碼 @encode(type) 獲取表示該類型的字符串葵礼,而不必硬編碼号阿。然后調(diào)用forwardInvocation處理并鸵,這一步的處理可以直接轉(zhuǎn)發(fā)給其它對象,即和第二步的效果一樣扔涧,沒有必要园担,故在這一步,會改變消息的內(nèi)容枯夜,比如增加參數(shù)弯汰,改變方法名稱。
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
if ([NSStringFromSelector(aSelector) isEqualToString:@"drink"])
{
NSLog(@"簽名 進(jìn)入forward");
return [NSMethodSignature signatureWithObjCTypes:"v@:*"];
}
return [super methodSignatureForSelector:aSelector];
}
- (void)forwardInvocation:(NSInvocation *)anInvocation
{
SEL sel = @selector(water:);
NSString *str = [NSString stringWithFormat:@"%s%s%s%s",@encode(void),@encode(id),@encode(SEL),@encode(char *)];
const char * cStr = [str cStringUsingEncoding:NSUTF8StringEncoding];//等同于"v@:*"
NSMethodSignature *signature = [NSMethodSignature signatureWithObjCTypes:cStr];
anInvocation = [NSInvocation invocationWithMethodSignature:signature];
[anInvocation setTarget:self];
[anInvocation setSelector:@selector(water:)];
NSString *water = @"喝水";
[anInvocation setArgument:&water atIndex:2];//第三個參數(shù)是字符串
if ([self respondsToSelector:sel])
{
[anInvocation invokeWithTarget:self];
return;
}else
{
Son *s = [Son new];
if ([s respondsToSelector:sel])
{
[anInvocation invokeWithTarget:s];
return;
}
}
[super forwardInvocation:anInvocation];
}
- (void)water:(NSString *)str
{
NSLog(@"完整消息轉(zhuǎn)發(fā) %@",str);
}
通過第三步完整消息轉(zhuǎn)發(fā)我們發(fā)現(xiàn)drink方法已經(jīng)轉(zhuǎn)發(fā)到了water:方法里湖雹,其中"v@:"里,v代表返回值是void咏闪,@代表響應(yīng)者是一個ID對象 ,:代表選擇子摔吏,代表字符串鸽嫂,我們這里water:方法改變了方法名稱纵装,增加了字符串參數(shù)。如果還是找不到對應(yīng)方法据某,則會沿著繼承樹查找橡娄,最終還是沒找到就會報doesNotRecognizeSelector錯誤程序掛掉。