美文网首页
objc_msgSend执行流程

objc_msgSend执行流程

作者: xxxxxxxxx_ios | 来源:发表于2018-12-26 19:28 被阅读4次

    oc对象(包括类对象和实例对象)调用方法,主要有3个步骤,分别是消息发送,动态方法解析,消息转发

    1.消息发送

    消息发送

    2.动态方法解析

    动态方法解析
    • 说明
      其中YMPerson类的.h文件中声明一个 -run方法,但.m文件没有进行方法的实现。如果直接调用YMPerson实例对象的run方法,会报找不到方法的崩溃。但在.m实现了+resolveInstanceMethod,并在其中动态添加了方法之后,实例对象会调用添加的方法。当调用类方法的时候,如果类方法没有实现,也可以通过实现+resolveClassMethod,添加相关类方法实现,就会调用添加的类方法。

    • 代码
      新建一个类
      .h

    #import <Foundation/Foundation.h>
    
    NS_ASSUME_NONNULL_BEGIN
    
    @interface YMPerson : NSObject
    - (void)run;
    @end
    
    NS_ASSUME_NONNULL_END
    
    

    .m

    #import "YMPerson.h"
    #import <objc/runtime.h>
    @implementation YMPerson
    + (BOOL)resolveInstanceMethod:(SEL)sel {
    
        if (sel == @selector(run)) {
            Method method = class_getInstanceMethod(self, @selector(test));
            class_addMethod(self, sel, method_getImplementation(method), method_getTypeEncoding(method));
            return YES;
        }
        return [super resolveInstanceMethod:sel];
    }
    - (void)test {
         NSLog(@"%s", __func__);
    }
    
    • 运行结果


      动态方法解析

    3.消息转发

    消息转发
    • 说明
      1.当动态方法解析阶段没有进行任何操作的时候,就会进入消息转发阶段,此阶段会首先调用forwardingTargetForSelector,返回值不为nil,进入消息发送阶段。如果返回值为nil,会调用methodSignatureForSelector。
      2.methodSignatureForSelector方法返回值为nil,调用doesNotRecognizeSelector,返回值不为nil,调用forwardInvocation方法。

    1.当调用forwardingTargetForSelector,会要求返回一个target,然后会去寻找这个target中对应的方法,这个方法必须要实现,不然报doesNotRecognizeSelector。当target中实现了指定的方法,会进行此target的消息发送。

    • 代码
      在刚才的代码基础上新建一个YMCat类,并实现-run方法。然后在YMPerson类中指定target,YMCat中的-run方法被调用,实现消息转发。

    新建YMCat类

    #import <Foundation/Foundation.h>
    
    NS_ASSUME_NONNULL_BEGIN
    
    @interface YMCat : NSObject
    - (void)run;
    @end
    
    NS_ASSUME_NONNULL_END
    
    #import "YMCat.h"
    
    @implementation YMCat
    - (void)run {
        
        NSLog(@"%s",__func__);
    }
    @end
    
    
    #import "YMPerson.h"
    #import <objc/runtime.h>
    #import "YMCat.h"
    @implementation YMPerson
    - (id)forwardingTargetForSelector:(SEL)aSelector {
        
        if (aSelector == @selector(run)) {
            
            return [[YMCat alloc] init];
        }
        return [super forwardingTargetForSelector:aSelector];
    }
    @end
    
    • 运行结果


      YMCat类run方法被调用

    2.当forwardingTargetForSelector返回值为nil,即没有指定target时,会进入消息转发的最后阶段,调用methodSignatureForSelector,此方法会要求返回一个方法签名,此方法签名包含了方法的返回值类型、参数类型,然后调用forwardInvocation,并携带一个NSInvocation对象,对此对象指定任务target,实现消息转发

    #import "YMPerson.h"
    #import <objc/runtime.h>
    #import "YMCat.h"
    @implementation YMPerson
    
    - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
        
        if (aSelector == @selector(run)) {
            
            // v @ : 分别b表示返回值为void,还包含-run的两个默认参数,id类型self, SEL类型_cmd
            return [NSMethodSignature signatureWithObjCTypes:"v@:"];
        }
        return [super methodSignatureForSelector:aSelector];
    }
    - (void)forwardInvocation:(NSInvocation *)anInvocation {
        
        /// 指定任务target 调用YMCat中的-run  从而实现消息转发
        [anInvocation invokeWithTarget:[[YMCat alloc] init]];
    }
    @end
    
    • 运行结果


      使用方法签名实现消息转发

    参考:MJ老师课程。

    相关文章

      网友评论

          本文标题:objc_msgSend执行流程

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