引入例子
@interface Person : NSObject
- (void)logString:(NSString *)str;
@end
@implementation Person
@end
Person
声明了一个方法但是没有实现。这时如果调用会崩溃
Person * p = [Person new];
[p logString:@"哈哈哈哈😀"];
消息转发的过程:
Isa去类对象中找到方法,然后去发送消息。找不到的话去父类一层一层往上。如果还没找到就会进到动态解析。
(注意:isa指向类;类对象的isa指向元类;如果当前类继承自NSObject,那么元类再指向根源类;如果当前类继承自NSObject,那么元类再指向自己)
动态解析流程:
// 拦截类方法
+ (BOOL)resolveClassMethod:(SEL)sel {
return NO;
}
// 拦截实例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
return NO;
}
以此为例,我们要在Person中拦截
@implementation Person
// 注意 `id self,SEL log`前两个参数必须要加上接收,不能只写str这一个参数
void dynamicLogString(id self,SEL log,NSString *str) { // 1.1
NSLog(@"%@", str);
}
// 拦截实例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
// 动态创建方法接收消息
if (sel == @selector(logString:)) {
class_addMethod(self, sel, (IMP)dynamicLogString, @"v@:@"); // 1.2
}
return NO;
}
@end
- 1.1
我们要引用的动态方法(dynamicLogString
为例),必须接收两个参数,如果需要接受额外参数,那么必须在后面追加。 - 1.2
参数v@:@
是根据原-logString:
方法的配置来定的v
对应void
;@
对应方法(即id类型);:
代表有参数;@
代表参数类型;
参数配置连接
以上就是第一步拦截,但是如果没有设置动态方法或者不想通过动态添加方法的方式处理的话,系统就进行动态转发
;
寻找备用接收者
- (id)forwardingTargetForSelector:(SEL)aSelector
设置一个合适的对象来调用这个方法(该对象声明+实现了该方法)
// 拦截实例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel {
// 动态创建方法接收消息
// 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];
}
如果还是没找到合适的对象来接收这个方法的话会进入方法转发
方法转发
分两步:
1.方法签名
2.方法转发
- (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字符串不是@"",不然会crash
}
return [super methodSignatureForSelector:aSelector];
}
// 方法转发
- (void)forwardInvocation:(NSInvocation *)anInvocation {
if ([TempObjct instancesRespondToSelector:anInvocation.selector]) {
[anInvocation invokeWithTarget:[TempObjct new]];
return;
}
// super
[super forwardInvocation:anInvocation];
}
终极拦截
如果以上方法都没有拦截到,或者想要保证程序不会崩溃,可以添加下面方法
- (void)doesNotRecognizeSelector:(SEL)aSelector {
NSLog(@"未知方法");
}
注意这个方法会捕获到最终所有没法直行的方法;
以上就是完整流程