引入例子
@interface Person : NSObject
- (void)logString:(NSString *)str;
@end
@implementation Person
@end
Person聲明了一個(gè)方法但是沒有實(shí)現(xiàn)。這時(shí)如果調(diào)用會(huì)崩潰
Person * p = [Person new];
[p logString:@"哈哈哈哈??"];
消息轉(zhuǎn)發(fā)的過程:
Isa去類對(duì)象中找到方法,然后去發(fā)送消息。找不到的話去父類一層一層往上。如果還沒找到就會(huì)進(jìn)到動(dòng)態(tài)解析。
(注意:isa指向類;類對(duì)象的isa指向元類;如果當(dāng)前類繼承自NSObject,那么元類再指向根源類;如果當(dāng)前類繼承自NSObject,那么元類再指向自己)
動(dòng)態(tài)解析流程:
// 攔截類方法
+ (BOOL)resolveClassMethod:(SEL)sel {
return NO;
}
// 攔截實(shí)例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
return NO;
}
以此為例,我們要在Person中攔截
@implementation Person
// 注意 `id self,SEL log`前兩個(gè)參數(shù)必須要加上接收,不能只寫str這一個(gè)參數(shù)
void dynamicLogString(id self,SEL log,NSString *str) { // 1.1
NSLog(@"%@", str);
}
// 攔截實(shí)例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
// 動(dòng)態(tài)創(chuàng)建方法接收消息
if (sel == @selector(logString:)) {
class_addMethod(self, sel, (IMP)dynamicLogString, @"v@:@"); // 1.2
}
return NO;
}
@end
- 1.1
我們要引用的動(dòng)態(tài)方法(dynamicLogString為例),必須接收兩個(gè)參數(shù),如果需要接受額外參數(shù),那么必須在后面追加。 - 1.2
參數(shù)v@:@是根據(jù)原-logString:方法的配置來定的v對(duì)應(yīng)void;@對(duì)應(yīng)方法(即id類型);:代表有參數(shù);@代表參數(shù)類型;
參數(shù)配置連接
以上就是第一步攔截,但是如果沒有設(shè)置動(dòng)態(tài)方法或者不想通過動(dòng)態(tài)添加方法的方式處理的話,系統(tǒng)就進(jìn)行動(dòng)態(tài)轉(zhuǎn)發(fā);
尋找備用接收者
- (id)forwardingTargetForSelector:(SEL)aSelector
設(shè)置一個(gè)合適的對(duì)象來調(diào)用這個(gè)方法(該對(duì)象聲明+實(shí)現(xiàn)了該方法)
// 攔截實(shí)例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
// 動(dòng)態(tài)創(chuàng)建方法接收消息
// if (sel == @selector(logString:)) {
// class_addMethod(self, sel, (IMP)dynamicLogString, @"v@:@");
// }
return NO;
}
- (id)forwardingTargetForSelector:(SEL)aSelector {
if ([TempObjct instancesRespondToSelector:aSelector]) {
return [TempObjct new];
}
return [super forwardingTargetForSelector:aSelector];
}
如果還是沒找到合適的對(duì)象來接收這個(gè)方法的話會(huì)進(jìn)入方法轉(zhuǎn)發(fā)
方法轉(zhuǎn)發(fā)
分兩步:
1.方法簽名
2.方法轉(zhuǎn)發(fā)
- (id)forwardingTargetForSelector:(SEL)aSelector {
// if ([TempObjct instancesRespondToSelector:aSelector]) {
// return [TempObjct new];
// }
return [super forwardingTargetForSelector:aSelector];
}
// 方法簽名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
if (aSelector == @selector(logString:)) {
return [NSMethodSignature signatureWithObjCTypes:"@@:*"]; // 注意這里接受的是c字符串不是@"",不然會(huì)crash
}
return [super methodSignatureForSelector:aSelector];
}
// 方法轉(zhuǎn)發(fā)
- (void)forwardInvocation:(NSInvocation *)anInvocation {
if ([TempObjct instancesRespondToSelector:anInvocation.selector]) {
[anInvocation invokeWithTarget:[TempObjct new]];
return;
}
// super
[super forwardInvocation:anInvocation];
}
終極攔截
如果以上方法都沒有攔截到,或者想要保證程序不會(huì)崩潰,可以添加下面方法
- (void)doesNotRecognizeSelector:(SEL)aSelector {
NSLog(@"未知方法");
}
注意這個(gè)方法會(huì)捕獲到最終所有沒法直行的方法;
以上就是完整流程