美文网首页
用代码理解iOS消息转发与SEL、IMP

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

作者: 差不多先生__ | 来源:发表于2018-04-24 23:13 被阅读0次

    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:

    图1.jpg
    这是为什么呢?再说为什么之前我们先看一下他传递进来了什么参数,传递结果如图2:
    图2.jpg
    传递进来的不是我们所写的“小明拿起了”,而是一个OC对象,其实在这里需要将函数补齐一下,因为它有两个隐藏参数,这两个参数分别是self_cmd,补齐后见代码
    void holdUpString(id self, SEL _cmd, NSString *string) {
        NSLog(@"%@砖头", string);
    }
    
    这次的打印结果见图3 图3.jpg

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

    图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

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