一、了解SEL前的准备-----isa指针简述
1、一个类就像一个 C 结构,NSObject 声明了一个成员变量: isa。由于 NSObject 是所有类的根类,所以所有的对象都会有一个 isa 的成员变量,而该 isa 变量指向该对象的类(空间)。
2、类在Objective-C中也是一个实体, 由于存在Objective-C 运行环境所有的类将有自己的存储空间。Objective-C 运行环境将为每个类分配空间。这里所说的 isa,正是指向这样一个类的空间, 从而建立类和对象之间的对应关系。
3、类空间(Class)包含了该类定义的成员变量,以及方法实现,,还包含了指向自己父类空间的指针。
4、Class里也有个isa的指针, 指向meteClass(元类)。元类保存了类方法的列表。当类方法被调用时,先会从本身查找类方法的实现,如果没有,元类会向他父类查找该方法。同时注意的是:元类(meteClass)也是类,它也是对象。元类也有isa指针,它的isa指针最终指向的是一个根元类(root meteClass)。根元类的isa指针指向本身,这样形成了一个封闭的内循环。
二、SEL 和 @selector()
1、selector 、 SEL 、@selector
(1)类中的方法以 selector 作为索引。selector 的数据类型是 SEL。
(2)虽然 SEL 定义成 char*(字符串),我们可以把它理解成 int。每个方法的名字对应一个唯一的 int 值。比如, 方法 addObject: 可能对应的是 12,当寻找该方法时,使用的是 selector,而不是名字 @"addObject:"
(3)Objective-C 数据结构中,存在一个 name - selector 的映射表如图 3.16
(4)SEL这个类型本质是类方法的编号(函数地址、函数指针、字符串、int、函数编号)
(5)可以理解 @selector() 就是取类方法的编号, 他的行为基本可以等同C语言的中函数指针, 只不过C语言中,可以把函数名直接赋给一个函数指针,而Object-C的类不能直接应用函数指针,这样只能做一个@selector语法来取。@selector的结果是SEL类型,所以说白了,@selector还是获得方法的编号(地址)。
小白总结:SEL其实是对方法的一种包装 , 将方法包装成一个SEL类型的数据(selector),去寻找对应的方法地址, 找到方法地址后就可以调用方法,这些都是运行时特性,发消息就是发送selector,然后根据selector找到地址,调用方法。 如果子类中找不到,就向上找,直到找到NSobject,没有就报错。SEL本身就是运行时特性的一个现实运用!
2、方法调用的过程及本质
(1)通俗理解 —— 对象调用方法时的寻找过程(寻找方法的过程): selector选择器机制
<1>首先把test这个方法名包装成SEL类型的数据(一个selector) <2>根据SEL数据找到对应的方法地址 <3>根据方法地址调用响应的方法 <4>注意: 在这个操作过程中有缓存,第一次找的时候是一个一个的找(应该是找到了代码区中,一条条找),消耗性能之后再用到的时候直接使用 (个人理解:应该是放到“类对象”的方法列表中了,这也就不用重复检索了)。
(2)大神理解 ——objec_msgSend()函数
<1> 在编译的时候, 只要有方法的调用, 编译器都会通过 selector 来查找,所以(假设addObject的selector为12),调用方法:[myObject addObject:yourObject];
<2>上面将会编译变成:
objc_msgSend(myObject, 12, yourObject);
这里,objec_msgSend()函数将会使用 myObjec 的 isa 指针来找到 myObject 的类空间结构并在类空间结构中查找 selector 12 所对应的方法.如果没有找到,那么将使用指向父类的指 针找到父类空间结构进行 selector 12 的查找. 如果仍然没有找到,就继续往父类的父类一 直找,直到找到为止, 如果到了根类 NSObject 中仍然找不到,将会抛出异常。
<3>通过上面,我们可以看到, 这是一个很动态的查找过程。类的结构可以在运行的时候改变,这样可以很 容易来进行功能扩展,Objective-C 语言是动态语言, 支持动态绑定。
3、SEL的应用举例:
调用方法(本身就是把方法包装的),也可以定义SEL类型的变量,然后把这个变量作为参数