美文网首页
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执行流程

    Runtime- objc_msgSend执行流程 1.消息发送 receiver是否为nil,如果是nil直...

  • Objective-C Rumtime-消息机制

    objc_msgSend执行流程 OC中的方法调用,其实都是转换为objc_msgSend函数的执行 objc_m...

  • 【iOS-RunTime系列三】objc_msgSend()

    objc_msgSend() OC的方法调用,消息机制,给方法调用者发送消息 objc_msgSend 的执行流程...

  • iOS 消息发送机制(前篇)

    前言 在OC中的方法调用,其实都是转换为objc_msgSend()函数的调用 objc_msgSend的执行流程...

  • iOS底层原理 - Runtime-02

    objc_msgSend执行流程 OC中的方法调用,其实都是转换为objc_msgSend函数的调用objc_ms...

  • Runtime(二)

    (三)、objc_msgSend执行流程 OC中的方法调用,其实都是转换为objc_msgSend函数的调用 ob...

  • runtime-消息机制

    OC方法调用,其实都是转换为objc_msgSend函数调用objc_msgSend的执行流程可以分为3大阶段:消...

  • Runtime消息传递机制

    先看下整个Class的底层结构 objc_msgSend的源码是汇编实现的 objc_msgSend的执行流程01...

  • iOS 常见面试题 -- Runtime

    OC中方法的调用 其实都是转化为 objc_msgSend函数的调用 objc_msgSend函数的执行流程可以分...

  • Runtime(二)

    objc_msgSend执行流程 OC中的方法调用,其实都是转换为objc_msgSend函数的调用 在底层会这么...

网友评论

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

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