博客地址:张飞的技术博客
在学习Runtime的过程中,有不少的知识点是平时难以接触到的。所以刚开始接触Runtime的时候会碰到好些不熟悉的小知识点。下面就随我扒一扒Runtime中那些需要知道的小知识点吧!
SEL
SEL
是一种类型,该类型表示对方法的一种封装,平时我们说的选择器selector就是它啦!Objective-C在编译时候呢,会依据每一个方法的名字、参数序列,生成一个唯一的整型标识(Int类型的地址),这个标识就是SEL。也就是说每一个SEL对应一个方法,根据一个SEL就能找到一个方法,然后对它进行调用等操作。其实我们也可以理解为SEL是一个方法的指针(实际上不是)。来看看它的真面目吧,在objc.h
的49行处可以发现:
/// An opaque type that represents a method selector.
typedef struct objc_selector *SEL;
这样看来SEL的本质是一个objc_selector
类型的结构体指针。那该如何生成一个SEL类型的数据呢?编译器提供的有选择器@selector,NSSelectorFromString()方法,sel_registerName函数。
SEL methodSel1 = @selector(method);
SEL methodSel2 = NSSelectorFromString(method);
```
>NOTE:只要方法名一样,那么这个方法的SEL就一样,不管同名的方法是有没有关系(继承),即使在不同的类方法名一样那么这个方法的SEL也是一样的。但是同一个类中不能存在同名方法(即使参数类型一样也不行,当然参数个数不一样可以),在不同的类里面是可以有同名方法的,不同类的实例对象执行相同的selector时,会在各自的方法列表中去根据selector去寻找自己对应的IMP。
本质上,SEL只是一个指向方法的指针(准确的说,只是一个根据方法名hash化了的KEY值,能唯一代表一个方法),它的存在只是为了加快方法的查询速度。
###IMP
上面讲了,根据SEL能寻找到函数,其实IMP才是真正指向函数的指针,它指向了函数实现的首地址。其实SEL就是为了查找方法的IMP的。取得IMP后,我们就获得了执行这个方法代码的入口点,此时,我们就可以像调用普通的C语言函数一样来使用这个函数指针了。
>NOTE:通过取得IMP,我们可以跳过Runtime的消息传递机制,直接执行IMP指向的函数实现,这样省去了Runtime消息传递过程中所做的一系列查找操作,会比直接向对象发送消息高效一些。
###Method
Method是一个结构体,用于表示类中定义的方法。实际上相当于在SEL和IMP之间作了一个映射。有了SEL,我们便可以找到对应的IMP,从而调用方法的实现代码。
>方法操作相关函数:
``` Objective-C
// 调用指定方法的实现
id method_invoke ( id receiver, Method m, ... );
// 调用返回一个数据结构的方法的实现
void method_invoke_stret ( id receiver, Method m, ... );
// 获取方法名
SEL method_getName ( Method m );
// 返回方法的实现
IMP method_getImplementation ( Method m );
// 获取描述方法参数和返回值类型的字符串
const char * method_getTypeEncoding ( Method m );
// 获取方法的返回值类型的字符串
char * method_copyReturnType ( Method m );
// 获取方法的指定位置参数的类型字符串
char * method_copyArgumentType ( Method m, unsigned int index );
// 通过引用返回方法的返回值类型字符串
void method_getReturnType ( Method m, char *dst, size_t dst_len );
// 返回方法的参数的个数
unsigned int method_getNumberOfArguments ( Method m );
// 通过引用返回方法指定位置参数的类型字符串
void method_getArgumentType ( Method m, unsigned int index, char *dst, size_t dst_len );
// 返回指定方法的方法描述结构体
struct objc_method_description * method_getDescription ( Method m );
// 设置方法的实现
IMP method_setImplementation ( Method m, IMP imp );
// 交换两个方法的实现
void method_exchangeImplementations ( Method m1, Method m2 );
```
上面这些函数在通过Runtime实现一些东西的时候或者在看别人的代码的时候或许能用到,只需要知道有这么一些函数就可以了。
###Class
Class在我的一篇文章中有详细的讲解,请移步[OC中的类是怎么来的?](),这里不在赘诉了。
###_cmd
通过_cmd这个关键字能获取当前方法的`SEL`。看个栗子吧:
```
- (void)testCmd {
SEL currentMethodSel = _cmd;
NSLog(@"currentSel is :%s",(char *)currentMethodSel);
}
```
上面的打印结果为:
```
currentSel is :testCmd
```
###objc_msgSend
在OC中,对方法的调用被称之为发送消息。其实我们对方法的调用最后都会被编译为下面的这种形式:
>对于函数:
```
[target method1];
[target method2:var];
```
>没有参数:
```
objc_msgSend(target,@selector(method1));
```
>有参数:
```
objc_msgSend(target,@selector(method2:),var);
```
方法的调用过程在我的上一篇文章[OC中的类是怎么来的?]()
###结尾
有人说Runtime是OC的本质和精华,虽然我们在项目中不经常用,但是理解还是有很大的用处的。喜欢我的文章就给我鼓励吧!
![](http://upload-images.jianshu.io/upload_images/465386-2f3e022157448deb.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/200)