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:
这是为什么呢?再说为什么之前我们先看一下他传递进来了什么参数,传递结果如图2:
传递进来的不是我们所写的“小明拿起了”,而是一个OC对象,其实在这里需要将函数补齐一下,因为它有两个隐藏参数,这两个参数分别是
self
和_cmd
,补齐后见代码
void holdUpString(id self, SEL _cmd, NSString *string) {
NSLog(@"%@砖头", string);
}
这次的打印结果见图3为什么加了这两个隐藏参数就可以了呢?OC方法的调用会传递两个隐藏参数,self
是方法的调用者,这里的_cmd
是SEL类型,也就是方法编号。因为OC的方法调用其实就是消息发送,这个可以用代码查看吗?——答案是可以,方法如下,首先我们先在控制器内添加头文件#import <objc/message.h>
,然后在项目的build settings里面将Enable Strict Checking of objc_msgSend Calls置为NO详见图4
然后将控制器内的代码[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:), @"小明拿起了");
代码.
详细代码点击传送门