Runtime- objc_msgSend执行流程

2021-03-15  本文已影响0人  我是卖报的小行家

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];
上一篇 下一篇

猜你喜欢

热点阅读