1.self 和 super 区别
self:代表当前方法的调用者,可以是类也可以是实例对象
super:编译器指示符
调用self init方法实际是调用objc_msgSend(self,@selecter(init));
调用super init方法实际是调用了objc_msgSendSuper(objc_super*,@selecter(init));
具体过程应该是:首先构建objc_super的结构体,结构体中两个参数:receiver(当前类的实例)和superClass(当前类的父类),系统会从superClass中查找init方法,获取指针,然后给用objc_super->receiver调用该方法;相当于执行objc_msgSend(self,@selecter(init)),只是方法init的指针指向superClass的init方法
因此打印NSStringFromClass([self class])和NSStringFromClass([super class])结果是一样的
2.KVO实现原理
实现方法:[objectA addObserver:self forKeyPath:@"name" NSKeyValueObservingOptionNew context:nil];
回调方法:-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
系统内部过程:
1. 当对objectA添加观察者时,动态创建其子类:NSKVONotifying_ObjectA(可在addObserver之后用object_getClass(objectA)打印)
2. 对于被观察的属性,重写set方法,添加:- willChangeValueForKey 和 -didChangeValueForKey通知观察者
3. 当移除观察时,动态移除创建的子类
好处:重写set方法时也可以监听
也可自己实现:- (BOOL)automaticallyNotifiesObserversForKey {return NO} 再重写set方法加入- willChangeValueForKey 和 -didChangeValueForKey即可
3. isa指针相关---动态创建类
isa:是一个Class类的指针
实例对象:1. isa指针指向该实例对应的类
类对象: 1. 包含实例对象的成员变量已经实例方法,协议等 2. isa指针指向元类
元类对象:1. 包含类方法,当未找到相应方法时,会像她的父类查找该方法 2.也有isa指针,它的isa指针最终指向的是一个根元类 root metaClass 3. 根元类的isa指向指向自己
注:1.object_getClass跟随实例的isa指针,返回此实例所属的类,对于实例对象(instance)返回的是类(class),对于类(class)则返回的是元类(metaClass) 2.-class方法对于实例对象(instance)会返回类(class),但对于类(class)则不会返回元类(metaClass),而只会返回类本身 3.class_isMetaClass可判断某类是否为元类.
扩展:使用objc_allocateClassPair可在运行时创建新的类与元类对,使用class_addMethod和class_addIvar可向类中增加方法和实例变量,最后使用objc_registerClassPair注册后,就可以使用此类了,用完用objc_disposeClassPair可销毁。(可用于KVO)