GNU 中 KVO isa swizzling具体流程

记录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]);
  1. 主要方法在于 利用运行时 objc_allocateClassPair、objc_registerClassPair动态创建一个原类的子类;
  2. GSObjCAddClassBehavior(replacement, baseClass); 将baseClass中的方法拷贝到新创建的类中。(这里的baseClass,是GSKVOBase类,之前一直看错,以为是原始类,所以后面完全理解不了。)
  3. object_setClass 修改对象所属的类 为新创建的类
GSKVOBase 实现:
- (Class) class
{
  return class_getSuperclass(object_getClass(self));
}
- (Class) superclass
{
  return class_getSuperclass(class_getSuperclass(object_getClass(self)));
}
  1. 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);
}
  1. GSKVOSetter 中默认实现了不同类型setter方法。内部实现基本相同(还实现了KVC相关内容)
  2. 在添加监听者的时候会调用- (void) overrideSetterFor: (NSString*)aKey方法,根据key 找到对应的setter方法,然后根据类型去获取GSKVOSetter类中相对应数据类型的setter方法,然后class_addMethod(replacement, sel, imp, [sig methodType])添加到对象所属的新类中。
  3. 此时调用对象的setter方法遍会走到 GSKVOSetter 的setter实现中。
  4. GSKVOSetter 的setter实现中,主要内容就是在原方法调用前调用
      // pre setting code here
      [self willChangeValueForKey: key];
      (*imp)(self, _cmd, val);
      // post setting code here
      [self didChangeValueForKey: key];
  1. 其中imp是通过[[self class] instanceMethodForSelector: _cmd]原方法的实现。因为新类中重写的class方法,所以这里的[self class]获取的还是原始类。而且最重要的,虽然原setter方法被替换为新的实现,但是_cmd获取的还是原方法名。所以instanceMethodForSelector会获取原始类中的setter方法。

你可能感兴趣的:(GNU 中 KVO isa swizzling具体流程)