1.category和原始类中的+(void)load;的调用顺序
先说结论,调用顺序是先调用原始类的load方法,再去调用category的load方法。为什么呢?
//person class
#import "Person.h"
#import "Person+A.h"
#import "Person+B.h"
@implementation Person
+ (void)load {
[Person test];
NSLog(@"load person");
}
+ (void)test {
NSLog(@"wakakak");
}
@end
//Person+A class
#import "Person+A.h"
@implementation Person (A)
+ (void)load {
NSLog(@"load person A");
}
+ (void)test {
NSLog(@"wo shi A");
}
@end
打印结果:
wo shi A
load person
load person B
load person A
结果却是让我疑惑不解,按理说category的方法“覆盖”原始类方法是基本常识,但load方法却全部的类都调用了。这是为啥?
大神的扒皮过程 分类的本质
runtime的源码入手:
1.找到objc-os.mm这个文件,然后找到这个文件的void _objc_init(void)这个方法,runtime的初始化都是在这个方法里面完成。
2.这个方法的最后一行调用了函数_dyld_objc_notify_register(&map_images, load_images, unmap_image);,我们点进load_images,这是加载模块的意思。
那么这样我们就搞清楚了为什么load方法不是像test方法一样,执行分类的实现
因为load方法的调用并不是objc_msgSend机制,它是直接找到类的load方法的地址,然后调用类的load方法,然后再找到分类的load方法的地址,再去调用它。所以看起来,他们的load方法都被调用了。但其它的方法,例如分裂和父类都写了test方法,并且在父类里导入分类,那么这时候调用情况是什么?
而test方法是通过消息机制去调用的。首先找到类对象,由于test方法是类方法,存储在元类对象中,所以通过类对象的isa指针找到元类对象,然后在元类对象中寻找test方法,由于分类也实现了test方法,所以分类的test方法是在类的test方法的前面,首先找到了分类的test方法,然后去调用它。
总结:
1.先调用类的load方法
按照编译先后顺序调用(先编译,先调用)
调用子类的load方法之前会先调用父类的load方法
2.再调用分类的load方法
按照编译先后顺序,先编译,先调用。顺序依据build phases-》compile sources中分类的顺序决定先后。
一般性方法,如果分类和原始类中都有,那么根据objc_msgSend的原理,根据runtime的消息传递机制中的核心函数void objc_msgSend(id self,SEL cmd,...)来发送消息,先从当前类中查找调用的方法,若没有找到则继续从其父类中一层层往上找,那么对于category重写同一个方法,则在消息传递的过程中,会最先找到category中的方法并执行该方法。对于多个分类调用同一个方法,Xcode在运行时是根据buildPhases->Compile Sources里面的从上至下顺序编译的,编译时通过压栈的方式将多个分类压栈,根据后进先出的原则,后编译的会被先调用,(插入顶部添加,即[methodLists insertObject:category_method atIndex:0]; 所以objc_msgSend遍历方法列表查找SEL 对应的IMP时,会先找到分类重写的那个,调用执行)当objc_msgSend找到方法并调用之后,就不再继续传递消息,所以形成所谓的覆盖。