objective-C 的消息及消息转发机制
第一、基本概念
1、objc_class中method数据结构
typedef struct objc_method *Method;
typedef struct objc_ method {
SEL method_name;
char *method_types;
IMP method_imp;
};
2、SEL
typedef struct objc_selector *SEL;
它是一个指向 objc_selector 指针,表示方法的名字/签名。
3、IMP
typedef id (*IMP)(id, SEL, ...);
IMP 是一个函数指针,输入参数包含一个接收消息的对象id(self 指针), 调用方法的选标 SEL (方法名),以及不定个数的方法参数,并返回一个id
第二、消息调用的机制
编译器会将消息转换为对消息函数 objc_msgSend的调用,该消息函数做了动态绑定所需要的一切工作:
1,它首先找到 SEL 对应的方法实现 IMP。因为不同的类对同一方法可能会有不同的实现,所以找到的方法实现依赖于消息接收者的类型。
2, 然后将消息接收者对象(指向消息接收者对象的指针)以及方法中指定的参数传递给方法实现 IMP。
3, 最后,将方法实现的返回值作为该函数的返回值返回。
第三、根据SEL找到IMP的过程
1,首先去该类objc_class的方法 cache中查找,如果找到了就返回它;
2,如果没有找到,就去该类的方法列表中查找。如果在该类的方法列表中找到了,则将 IMP返回,并将它加入cache中缓存起来。
3,如果在该类的方法列表中没找到对应的 IMP,在通过该类结构中的 super_class指针在其父类结构的方法列表中去查找,直到在某个父类的方法列表中找到对应的IMP,返回它,并加入cache中;
4,如果在自身以及所有父类的方法列表中都没有找到对应的 IMP,则进入消息转发机制