用代码理解iOS消息转发与SEL、IMP

OC的方法调用,其实就是消息发送。

在写代码之前先说一下SEL于IMP,OC方法由SEL和IMP组成,SEL是方法编号,IMP是函数指针(方法实现),一个指向函数的指针。
通俗地讲,一本书的目录上有标题,标题就是SEL,所谓的方法编号,标题后面的页码就是IMP,那个页码就是IMP指向的地方,就是所谓的函数指针。

首先我们创建一个Person继承于NSObject的类,里面添加一个方法- (void)holdUpString;Person.m文件里面添加方法

+ (BOOL)resolveInstanceMethod:(SEL)sel {
    //添加hold up方法
    class_addMethod(self, sel, holdUpString, "");
    return [super resolveInstanceMethod:sel];
}

void holdUpString() {
    NSLog(@"棍子");
}

在这里解释一下+ (BOOL)resolveInstanceMethod:(SEL)sel方法,这个函数在运行时(runtime),没有找到SEL的IML时就会执行这个方法;
在这个方法里面可以使用class_addMethod(Class _Nullable cls, SEL _Nonnull name, IMP _Nonnull imp, const char * _Nullable types)添加一个函数方法,class_addMethod根据名字就能看出来,是给一个类添加方法,这里面有四个参数,给解释一下
第一个参数是给谁添加方法,在这里当然是给Person类添加,所以第一个参数填写self
第二个参数resolveInstanceMethod:(SEL)sel已经传递给你了,直接填写这个sel,在这里的含义就是方法编号;
第三个参数是‘函数指针’,在这里指向的就是一个函数,在这里的函数就是void holdUpString()所以填写holdUpString就可以了;
第四个参数是用来标识IMP函数实现的返回值与参数,暂时填写""。

完成上述代码后,在控制器里面创建初始化Person类,并且调用方法holdUpString,Xcode打印“棍子”

在这里咱们提出问题,如果我们实现方法

- (void)holdUpString {
    NSLog(@"111");
}

那么上述的代码还会调用吗?答案是不会,理由参考上述针对+ (BOOL)resolveInstanceMethod:(SEL)sel方法的解释。

话题延伸,现在我们需要在这个函数传递参数,将Person.h方法- (void)holdUpString;注释,新添带有参数的方法- (void)holdUpString:(NSString *)string;并在控制器里面调用

    Person *p = [[Person alloc] init];
    [p holdUpString:@"小明拿起了"];

Person.m文件holdUpString函数注释且将resolveInstanceMethod方法实现改成下述代码,并添加新的holdUpString函数

+ (BOOL)resolveInstanceMethod:(SEL)sel {
    //添加hold up方法
    class_addMethod(self, sel, (IMP)holdUpString, "");
    return [super resolveInstanceMethod:sel];
}

void holdUpString(NSString *string) {
    NSLog(@"%@砖头", string);
}

抛出问题,现在如果我们运行代码,会打印什么?

打印结果如图1:

用代码理解iOS消息转发与SEL、IMP_第1张图片
图1.jpg

这是为什么呢?再说为什么之前我们先看一下他传递进来了什么参数,传递结果如图2:
用代码理解iOS消息转发与SEL、IMP_第2张图片
图2.jpg

传递进来的不是我们所写的“小明拿起了”,而是一个OC对象,其实在这里需要将函数补齐一下,因为它有两个隐藏参数,这两个参数分别是 self_cmd,补齐后见代码

void holdUpString(id self, SEL _cmd, NSString *string) {
    NSLog(@"%@砖头", string);
}

这次的打印结果见图3
图3.jpg

为什么加了这两个隐藏参数就可以了呢?OC方法的调用会传递两个隐藏参数,self是方法的调用者,这里的_cmdSEL类型,也就是方法编号。因为OC的方法调用其实就是消息发送,这个可以用代码查看吗?——答案是可以,方法如下,首先我们先在控制器内添加头文件#import ,然后在项目的build settings里面将Enable Strict Checking of objc_msgSend Calls置为NO详见图4

用代码理解iOS消息转发与SEL、IMP_第3张图片
图4.jpg

然后将控制器内的代码[p holdUpString:@"小明拿起了"];注释,新添代码,objc_msgSend(p, @selector(holdUpString:), @"小明拿起了");在这里解释一下objc_msgSend(id _Nullable self, SEL _Nonnull op, ...)的几个参数,
第一个参数是给谁发送消息,在这里是给p发送消息,填写p就好了;
第二个参数是给哪个方法编号发送消息,这里是要给- (void)holdUpString:(NSString *)string;所以填写@selector(holdUpString:);
第三个参数是发送什么消息“小明拿起了”.
OC的方法调用其实就是消息发送我的理解就是这么来的,上面OC代码[p holdUpString:@"小明拿起了"];的底层其实就是这句objc_msgSend(p, @selector(holdUpString:), @"小明拿起了");代码.
详细代码点击传送门

你可能感兴趣的:(用代码理解iOS消息转发与SEL、IMP)