每个objc对象内都有这几部分:1.对象方法list 2.成员变量list 3.属性list 4.isa指针
首先,对象方法list runtime使用objc_msgSend(receiver, selector)对方法进行调用 receiver是方法的调用者, selector则是方法 @selector其实是方法名 在调用方法时在receiver中的method list中找到该方法然后再找到它的IMP(实现),如果找不到的话这就会发生unrecognized selector而导致crash。当然这时候也还有补救措施,这就是传说中的消息转发。首先第一次是在+resolveInstanceMethod:或者 +resolveClassMethod: 在这里我们可以添加一个函数调用,然后返回yes这样安全了。如果没有就进行fast forwarding实现-forwardingTargetForSelector:方法进行对象转移,也就是改变方法的调用者。如果还没有调用就-methodSignatureForSelector:通过这个函数进行获得参数,在这里进行签名NSInvocation并发送-forwardInvocation:,否则就会造成crash。对象方法列表存放在类对象中。
16.09.14补充
id objc_msgSend(id self, SEL op, ...) {
if (!self) return nil;
IMP imp = class_getMethodImplementation(self->isa, SEL op);
imp(self, op, ...); //调用这个函数,伪代码...
}
//查找IMP
IMP class_getMethodImplementation(Class cls, SEL sel) {
if (!cls || !sel) return nil;
IMP imp = lookUpImpOrNil(cls, sel);
if (!imp) return _objc_msgForward; //_objc_msgForward 用于消息转发
return imp;
}
IMP lookUpImpOrNil(Class cls, SEL sel) {
if (!cls->initialize()) {
_class_initialize(cls);
}
Class curClass = cls;
IMP imp = nil;
do { //先查缓存,缓存没有时重建,仍旧没有则向父类查询
if (!curClass) break;
if (!curClass->cache) fill_cache(cls, curClass);
imp = cache_getImp(curClass, sel);
if (imp) break;
} while (curClass = curClass->superclass);
return imp;
}
下一个就是成员变量list 我们一般申明变量的时候在前面加_,比如_var。在成员变量list中存放当前类的实例变量以及所有父类的实例变量。
然后是属性list 申明属性用@property 一般来说property=ivar+setter+getter。在申明属性后,编译器会自动访问这些属性所需方法,这就是autosynthesis。这是在编译器进行,所以编译器里看不见代码。假如我们要更改setter,getter的名字,就在申明时在填充如@property(getter=???,setter=set???)这样就可以更改默认的生成方法。假如我们要更改默认生成的成员变量名字就采用@synthesize,如@synthesize darling = _myDarling。同时还有一个@dynamic,它会告诉编译器setter getter用户自己实现,如果没有实现这两个方法时正常编译是没问题,就是在使用这个变量时就会crash。
isa指针在第一次看挺玄学的,经常看见但不太了解什么作用。先放一张图。
众所周知,每个类在调用之前会生成一个静态对象。一个类存放着它的superclass指针指向父类,而isa指针 对象指向类对象 类对象的isa指针指向元对象,元对象的isa指针指向根元对象根类对象的superclass指向nil,根对象在objc中为nsobject。在调用一个方法时,通过isa指针找到对象所属的类,然后通过类的方法列表以及其父类的方法列表找寻方法进行调用,在进行objc_msgSend时并没有返回值,返回值都是具体的调用产生的,所以objc可以对nil进行方法调用不会crash,是因为寻找对象时isa指针就得到0返回值。元对象内部存放着类方法列表,而对象方法列表存储在类对象中。
那么类方法和实例方法有何不同?类方法是属于类对象的,所以只能给类对象调用,self也是指类对象,不能调用对象方法。实例方法是属于实例对象,只能通过实例对象调用,self就是指实例对象,也可以调用类方法。
通过短暂了解对象的各种姿势,我们来举个栗子。
self = [super init];
这在重写对象初始化时都会调用,但我们这时候调用[self class]和[super class]得出的结果会一样嘛?答案是一样的。原因是根据super是一个magic keyword,它是个编译器标示符和self指向的是同一个对象接受者。super的方法调用的函数是objc_msgSendSuper(struct objc_super *super, SEL op, ...)
struct objc_super {
__unsafe_unretained id receiver;
__unsafe_unretained Class super_class;
};
第一个receiver类似objc_msgSend的receiver,第二个参数是当前类的父类。在[self class]中objc_msgSend中self查找,没有,最后通过父类查找在nsobject中找到得到当前类。而[super class]首先构造结构体objc_super,第一个为self,所以查找到父类,然后在父类的方法列表中查找class,最后还是在nsobject的class中找到方法,而处理的对象还是self,所以两者的结果相同~有点绕
如果有错漏,请一定告诉我,学习还在继续!THX