美文网首页iOS源码解析
GNU 中 KVO isa swizzling具体流程

GNU 中 KVO isa swizzling具体流程

作者: 哦呵呵y | 来源:发表于2018-04-20 13:51 被阅读18次

记录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具体流程

    记录GNU中,KVO进行类替换的具体实现 一、创建一个用来替换的类 主要方法在于 利用运行时 objc_alloc...

  • 问题:KVO的底层实现原理

    KVO是基于runtime机制实现的,KVO运用了一个isa-swizzling技术。isa-swizzling就...

  • 问题:apple用什么方式实现对一个对象的KVO?

    KVO是基于runtime机制实现的,KVO运用了一个isa-swizzling技术。isa-swizzling就...

  • Runtime面试题

    1.oc消息转发流程 2.KVO原理KVO是基于runtime实现的,KVO运用了isa-swizzling将两个...

  • KVO底层原理分析

    一、 KVO内部实现原理 KVO是基于 runtime机制实现的,使用了isa 混写(isa-swizzling)...

  • KVO

    Apple 使用了 isa 混写(isa-swizzling)来实现 KVO 。当观察对象A时,KVO机制动态创建...

  • KVO和KVC

    什么是KVO? KVO是对OC对于观察者设计模式的实现 apple使用了isa混写(isa-swizzling)来...

  • KVO (Key-value observing)

    KVO是观察者模式的另一实现。 使用了isa混写(isa-swizzling)来实现KVO 使用setter方法改...

  • KVO 原理

    概念 KVO 是 OC 对观察者模式的一种实现 Apple 使用了 isa混写技术(isa - swizzling...

  • KVO实现原理

    KVO是基于runtime机制实现的,使用了isa-swizzling技术(isa替换) 下面是一个例子用来证明在...

网友评论

    本文标题:GNU 中 KVO isa swizzling具体流程

    本文链接:https://www.haomeiwen.com/subject/eebakftx.html