建了一个面试题解答的项目,大家可以看一看,希望大家帮忙给一个star,谢谢了! 项目地址:https://github.com/NotFound9/interviewGuide
起因
最近在复习iOS中的消息转发机制,如果需要在动态方法解析这一阶段对消息进行处理,一般需要调用class_addMethod方法给类动态地增加方法,例如:
我当时敲代码的时候发现自己对于class_addMethod这个方法的第四个参数const char *types不太清楚,后来搜了一下类型编码,发现网上的文章主要是对于属性的类型编码进行了介绍,也没有人对函数的类型编码进行介绍,就去看了一下文档了解了一下。再后来发现介绍消息转发机制的那篇文章中对于这个参数的传入是错误的,
图中添加的方法是void functionForMethod1(idself, SEL _cmd),
传入的函数类型编码是"@:",
而正确的函数类型编码应该是"v@:",
探究
下面让我们来看看这个方法:
BOOL class_addMethod(Class cls, SEL name, IMP imp, const char *types);
这个方法主要接受四个参数
Class cls 要添加方法的类
SEL name 被添加方法的名字
IMP imp 添加的方法的实现
const char *types 描述方法参数类型的字符数组。
描述方法参数类型的字符数组的第一个字符是代表返回值的类型,后面的字符依次代表参数的类型,因为Objective-C中的函数会包含两个隐式参数,也就是方法调用者和方法名,例如
+(void)method
实际应该是
void method(id self, SEL _cmd)
如果返回值为空,那么函数的类型编码的第一个字符是v,如果不为空,则为返回值类型对应的编码,详细的可以看下面的编码对应表
因为第一个参数是方法调用者,它的类型肯定是对象类型,所以类型编码的第二个字符一定是@
因为第二个参数是方法名的类型,第三个字符一定是:
所以这个函数
void method(id self, SEL _cmd)
的类型编码为 "v@:"
那么如果要添加的函数是一个set函数,类型编码是怎么样的呢?
-(void)setA:(NSString *)a
同理,set方法实际的函数是这样的:
void setA(id self, SEL _cmd, id a)
与上面无参数的方法相比,只是多了一个参数,
所以类型编码为"v@:@",代码为
class_addMethod(self, @selector(setA:), (IMP)setA,"v@:@");
悔悟。。。
最后发现自己还是太年轻了。。。
苹果还是考虑到了开发者手动写函数的类型编码容易出错的情况,所以还是提供了api来获取函数的类型编码(其实我想说。。。既然能提供api获取函数的类型编码,为什么不在class_addMethod的实现中去获取函数的类型编码,而需要外部传入)
主要是有这么一个函数
const char * _Nullablemethod_getTypeEncoding(Method _Nonnull m)
通过传入类名和,获函数名,获取函数的类型编码,例如这样: