objc_msgSend执行流程

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

1.消息发送

消息发送

2.动态方法解析

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

  • 代码
    新建一个类
    .h

#import 

NS_ASSUME_NONNULL_BEGIN

@interface YMPerson : NSObject
- (void)run;
@end

NS_ASSUME_NONNULL_END

.m

#import "YMPerson.h"
#import 
@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 

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 
#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 
#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执行流程)