美文网首页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具体流程

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