记录GNU中,KVO进行类替换的具体实现
一、创建一个用来替换的类
/*
* Create subclass of the original, and override some methods
* with implementations from our abstract base class.
*/
superName = NSStringFromClass(original);
name = [@"GSKVO" stringByAppendingString: superName];
template = GSObjCMakeClass(name, superName, nil);
GSObjCAddClasses([NSArray arrayWithObject: template]);
replacement = NSClassFromString(name);
GSObjCAddClassBehavior(replacement, baseClass);
...
object_setClass(self, [r replacement]);
- 主要方法在于 利用运行时
objc_allocateClassPair、objc_registerClassPair
动态创建一个原类的子类; -
GSObjCAddClassBehavior(replacement, baseClass);
将baseClass中的方法拷贝到新创建的类中。(这里的baseClass,是GSKVOBase类,之前一直看错,以为是原始类,所以后面完全理解不了。) -
object_setClass
修改对象所属的类 为新创建的类
GSKVOBase 实现:
- (Class) class
{
return class_getSuperclass(object_getClass(self));
}
- (Class) superclass
{
return class_getSuperclass(class_getSuperclass(object_getClass(self)));
}
- GSKVOBase类中重写了 class, superclass方法。所以原对象虽然已经属于新类,但是外部调用这两个方法仍返回原来的类。
二、重写set方法
@interface GSKVOSetter : NSObject
- (void) setter: (void*)val;
- (void) setterChar: (unsigned char)val;
- (void) setterDouble: (double)val;
- (void) setterFloat: (float)val;
- (void) setterInt: (unsigned int)val;
- (void) setterLong: (unsigned long)val;
#ifdef _C_LNG_LNG
- (void) setterLongLong: (unsigned long long)val;
#endif
- (void) setterShort: (unsigned short)val;
- (void) setterRange: (NSRange)val;
- (void) setterPoint: (NSPoint)val;
- (void) setterSize: (NSSize)val;
- (void) setterRect: (NSRect)rect;
@end
...
setter方法内部实现
- (void) setter: (void*)val
{
NSString *key;
Class c = [self class];
void (*imp)(id,SEL,void*);
imp = (void (*)(id,SEL,void*))[c instanceMethodForSelector: _cmd];
key = newKey(_cmd);
if ([c automaticallyNotifiesObserversForKey: key] == YES)
{
// pre setting code here
[self willChangeValueForKey: key];
(*imp)(self, _cmd, val);
// post setting code here
[self didChangeValueForKey: key];
}
else
{
(*imp)(self, _cmd, val);
}
RELEASE(key);
}
- GSKVOSetter 中默认实现了不同类型setter方法。内部实现基本相同(还实现了KVC相关内容)
- 在添加监听者的时候会调用
- (void) overrideSetterFor: (NSString*)aKey
方法,根据key 找到对应的setter方法,然后根据类型去获取GSKVOSetter类中相对应数据类型的setter方法,然后class_addMethod(replacement, sel, imp, [sig methodType])
添加到对象所属的新类中。 - 此时调用对象的setter方法遍会走到 GSKVOSetter 的setter实现中。
- GSKVOSetter 的setter实现中,主要内容就是在原方法调用前调用
// pre setting code here
[self willChangeValueForKey: key];
(*imp)(self, _cmd, val);
// post setting code here
[self didChangeValueForKey: key];
- 其中imp是通过
[[self class] instanceMethodForSelector: _cmd]
原方法的实现。因为新类中重写的class方法,所以这里的[self class]
获取的还是原始类。而且最重要的,虽然原setter方法被替换为新的实现,但是_cmd
获取的还是原方法名。所以instanceMethodForSelector
会获取原始类中的setter方法。