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方法。
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,856评论 0 9
  • 主要参考链接: http://yulingtianxia.com/blog/2014/11/05/objectiv...
    Kevin_Junbaozi阅读 3,386评论 0 10
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,269评论 19 139
  • 一直不明白父亲为什么在我这里住不惯。每次进城,他最多呆上十天半月,是决意要回去的。有几次我试图阻拦,都被他坚决地推...
    八里山人程远河阅读 507评论 5 9
  • 在你采撷果实的时候 我是被遗忘的花瓣 亦如麦田把黄色给了秋天 镰刀割短海浪,取流星之焰 湮没潮去的沙滩 盛开而来 ...
    飞狐119阅读 267评论 6 3