Runtime- objc_msgSend执行流程
1.消息发送
receiver是否为nil,如果是nil直接退出,如果不为空,通过receiverClass的cache中查找方法,如找到了则调用方法结束查找,如果未找到则从receiverClass的class_rw_t中的方法数组里遍历查找(如果方法列表里的方法是排好序的按照二分法,否则就是普通遍历挨个查找)方法,如果找到了方法,调用方法,结束查找,并且将方法缓存到receiverClass的cache中,如果还没有找到方法,则从superclass的cache中查找,如果找到,则调用方法,结束查找,并且将方法缓存到receiverClass的cache中,如果父类缓存还未找到,则从superclass的class_rw__t中的方法数组里遍历查找,如果找到,则调用方法,结束查找,并且将方法缓存到receiverClass的cache中,如果还没找到继续找superclass,(cache/superclass_rw_t),如果还是没有,则进入到第二个阶段,消息动态解析阶段
2.动态方法解析
先判断是否曾经有过动态解析,如果没有则调用+resolveInstanceMethod:或者+resolveClassMethod:方法来动态解析方法,然后再标记为已经动态解析,完成后在进行消息发送
动态解析过后会重新走“消息发送的流程”,从“reserveClass的cache中查找方法”这一步开始执行
如果还是找不到方法会再次进入动态方法解析,此时因为已经之前有过动态解析,所以会进入到第三阶段消息转发。
//对象方法
-(void)other{
NSLog(@"%s",__func__);
}
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
if (sel == @selector(study)) {
Method method = class_getInstanceMethod(self, @selector(other));
class_addMethod(self, sel, method_getImplementation(method), method_getTypeEncoding(method));
}
return [super resolveInstanceMethod:sel];
}
//类方法
+(void)other2{
NSLog(@"%s",__func__);
}
+ (BOOL)resolveClassMethod:(SEL)sel
{
if (sel == @selector(work)) {
Method method = class_getClassMethod(object_getClass(self), @selector(other2));
class_addMethod(object_getClass(self), sel, method_getImplementation(method), method_getTypeEncoding(method));
return YES;
}
return [super resolveClassMethod:sel];
}
3.消息转发
如果步骤2没有执行则进入消息转发阶段,调用forwardingTargetForSelector,如果返回值不为nil,objc_msgSend(返回值,SEL),如果返回值为nil,则调用methodsignatureForSelector:方法,如果调用这个方法后返回值还是为nil,则会调用doseNotRecognizeSelector:方法报经典错误。而如果返回值不为空(方法签名)则会调用forwardInvocation:方法,那就可以尽情的在这个方法里做想做的事情了。
//forwardingTargetForSelector返回值不为nil,转发给别的有这个方法的对象执行
-(id)forwardingTargetForSelector:(SEL)aSelector
{
if (aSelector == @selector(study)) {
//消息转发,由MJCat对象代替其方法,前提MJCat也有study方法
return [[MJCat alloc]init];
}
return [super forwardingTargetForSelector:aSelector];
}
//如果返回值为nil,forwardInvocation
//返回方法签名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
if (aSelector == @selector(study)) {
//v16@0:8 = void xxx (self,_cmd)
return [NSMethodSignature signatureWithObjCTypes:"v16@0:8"];
}
return [super methodSignatureForSelector:aSelector];
}
//NSInvocation - 方法调用
//方法名 - anInvocation.selector
//方法调用 - anInvocation.target
//方法参数 - anInvocation getArgument: atIndex:
- (void)forwardInvocation:(NSInvocation *)anInvocation{
//设置方法调用者
[anInvocation invokeWithTarget:[[MJCat alloc]init]];
}
补充:
类方法也可以实现消息转发,但是用的是`+ (id)forwardingTargetForSelector:(SEL)aSelector`函数//类方法的消息转发,
[MJStudent test];