美文网首页
Runtime- objc_msgSend执行流程

Runtime- objc_msgSend执行流程

作者: 我是卖报的小行家 | 来源:发表于2021-03-15 10:34 被阅读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];
    

    相关文章

      网友评论

          本文标题:Runtime- objc_msgSend执行流程

          本文链接:https://www.haomeiwen.com/subject/rwzhcltx.html