KVC

本文章的代碼來源gnustep-base-1.26.0 ,應(yīng)該和系統(tǒng)的實(shí)現(xiàn)還是不一樣的,僅供參考 :

常用方法

+ (BOOL) accessInstanceVariablesDirectly; // 是否允許訪問成員變量,默認(rèn)是yes

- (void) setValue: (id)anObject forKey: (NSString*)aKey;
- (void) setValue: (id)anObject forKeyPath: (NSString*)aKey;
- (void) setValue: (id)anObject forUndefinedKey: (NSString*)aKey;

/**
 * Returns the value associated with the supplied key as an object.
 * Scalar attributes are converted to corresponding objects.<br />
 * The search order is:<br/>
 * Accessor methods:
 * <list>
 *  <item>getKey</item>
 *  <item>key</item>
 * </list>
 * If the receiver's class allows +accessInstanceVariablesDirectly
 * it continues with private accessors:
 * <list>
 *  <item>_getKey</item>
 *  <item>_key</item>
 * </list>
 * and then instance variables:
 * <list>
 *  <item>key</item>
 *  <item>_key</item>
 * </list>
 * Invokes -setValue:forUndefinedKey:
 * if no accessor mechanism can be found
 * and raises NSInvalidArgumentException if the accessor method takes
 * any arguments or the type is unsupported (e.g. structs).
 */
- (id) valueForKey: (NSString*)aKey;
- (id) valueForKeyPath: (NSString*)aKey;
- (id) valueForUndefinedKey: (NSString*)aKey;

setValue: forKey:

- (void) setValue: (id)anObject forKey: (NSString*)aKey
{
  unsigned  size = [aKey length] * 8; // 乘以8是為了兼容編碼長度問題
  char      key[size + 1]; // +1 是為了轉(zhuǎn)成的字符串最后有個(gè)'/0'終止符
#ifdef WANT_DEPRECATED_KVC_COMPAT //  要廢棄,但是要兼容
  IMP       o = [self methodForSelector: @selector(takeValue:forKey:)];

  setupCompat();
  if (o != takeValue && o != takeValueKVO)
    {
      (*o)(self, @selector(takeValue:forKey:), anObject, aKey); // 調(diào)用takeValue:forKey:方法
      return;
    }
#endif

  [aKey getCString: key
     maxLength: size + 1
      encoding: NSUTF8StringEncoding]; // 轉(zhuǎn)成 char * 字符串
  size = strlen(key);
  SetValueForKey(self, anObject, key, size); // 調(diào)用 SetValueForKey 函數(shù)
}

可以看到,調(diào)用了takeValue:forKey方法或者SetValueForKey函數(shù)

takeValue:forKey:

- (void) takeValue: (id)anObject forKey: (NSString*)aKey
{
  SEL       sel = 0;
  const char    *type = 0;
  int       off = 0;
  unsigned  size = [aKey length] * 8;
  char      key[size + 1];

  GSOnceMLog(@"This method is deprecated, use -setValue:forKey:"); // 方法廢棄
  [aKey getCString: key
     maxLength: size + 1
      encoding: NSUTF8StringEncoding]; // 轉(zhuǎn)成C語言字符串, 目測接下來的實(shí)現(xiàn)會(huì)和SetValueForKey一樣
  size = strlen(key); // 獲得轉(zhuǎn)換后的char*字符串長度

  if (size > 0)
    {
      const char    *name;
      char      buf[size + 6];
      char      lo;
      char      hi;

      strncpy(buf, "_set", 4);
      strncpy(&buf[4], key, size); // 拼接成_setKey的格式
      lo = buf[4];
      hi = islower(lo) ? toupper(lo) : lo;// 如果首字母是小寫,轉(zhuǎn)成大寫,駝峰命名法
      buf[4] = hi;
      buf[size + 4] = ':'; // 轉(zhuǎn)成方法 
      buf[size + 5] = '\0';

      name = &buf[1];   // setKey:
      type = NULL;
      sel = sel_getUid(name); // 尋找或注冊方法選擇器
      if (sel == 0 || [self respondsToSelector: sel] == NO) // 如果沒有實(shí)現(xiàn) setKey方法,就去尋找_setKey方法
    {
      name = buf;   // _setKey:
      sel = sel_getUid(name);
      if (sel == 0 || [self respondsToSelector: sel] == NO)
        {
          sel = 0;
          if ([[self class] accessInstanceVariablesDirectly] == YES) // 是否能訪問成員變量
        {
          buf[size + 4] = '\0';
          buf[3] = '_';
          buf[4] = lo;
          name = &buf[4];   // 尋找key成員變量
          if (GSObjCFindVariable(self, name, &type, &size, &off) == NO)
            {
              name = &buf[3];   // 尋找_key 成員變量
              GSObjCFindVariable(self, name, &type, &size, &off);
            }
        }
        }
    }
    }
  GSObjCSetVal(self, key, anObject, sel, type, size, off);
}

SetValueForKey

static void
SetValueForKey(NSObject *self, id anObject, const char *key, unsigned size)
{
  SEL       sel = 0;
  const char    *type = 0;
  int       off = 0;

  if (size > 0)
    {
      const char    *name;
      char      buf[size + 6];
      char      lo;
      char      hi;

      strncpy(buf, "_set", 4);
      strncpy(&buf[4], key, size);
      lo = buf[4];
      hi = islower(lo) ? toupper(lo) : lo;
      buf[4] = hi;
      buf[size + 4] = ':';
      buf[size + 5] = '\0';

      name = &buf[1];   // setKey:
      type = NULL;
      sel = sel_getUid(name);
      if (sel == 0 || [self respondsToSelector: sel] == NO)
    {
      name = buf;   // _setKey:
      sel = sel_getUid(name);
      if (sel == 0 || [self respondsToSelector: sel] == NO)
        {
          sel = 0;
          if ([[self class] accessInstanceVariablesDirectly] == YES)
        {
          buf[size + 4] = '\0';
          buf[3] = '_';
          buf[4] = lo;
          name = &buf[3];   // _key
          if (GSObjCFindVariable(self, name, &type, &size, &off) == NO)
            {
              buf[4] = hi;
              buf[3] = 's';
              buf[2] = 'i';
              buf[1] = '_';
              name = &buf[1];   // _isKey
              if (GSObjCFindVariable(self,
            name, &type, &size, &off) == NO)
            {
              buf[4] = lo;
              name = &buf[4];   // key
              if (GSObjCFindVariable(self,
                name, &type, &size, &off) == NO)
                {
                  buf[4] = hi;
                  buf[3] = 's';
                  buf[2] = 'i';
                  name = &buf[2];   // isKey
                  GSObjCFindVariable(self,
                name, &type, &size, &off);
                }
            }
            }
        }
        }
      else
        {
          GSOnceFLog(@"Key-value access using _setKey: is deprecated:");
        }
    }
    }
  GSObjCSetVal(self, key, anObject, sel, type, size, off);
}

- (void) takeValue: (id)anObject forKey: (NSString*)aKey 實(shí)現(xiàn)基本上一致,只是在訪問成員變量的時(shí)候,會(huì)再檢索 _isKeyisKey

GSObjCFindVariable 尋找成員變量

BOOL
GSObjCFindVariable(id obj, const char *name,
  const char **type, unsigned int *size, int *offset)
{
  Class     class = object_getClass(obj);
  Ivar      ivar = class_getInstanceVariable(class, name); // 獲取成員變量

  if (ivar == 0)
    {
      return NO;
    }
  else
    {
      const char    *enc = ivar_getTypeEncoding(ivar); // 獲取類型

      if (type != 0)
    {
      *type = enc;
    }
      if (size != 0)
    {
      NSUInteger    s;
      NSUInteger    a;

      NSGetSizeAndAlignment(enc, &s, &a);
      *size = s; // 獲取成員變量size
    }
      if (offset != 0)
    {
      *offset = ivar_getOffset(ivar); // 成員變量的偏移量
    }
      return YES;
    }
}

GSObjCSetVal 調(diào)用setter方法或者直接修改成員變量

void
GSObjCSetVal(NSObject *self, const char *key, id val, SEL sel,
  const char *type, unsigned size, int offset)
{
  static NSNull     *null = nil;
  NSMethodSignature *sig = nil;

  if (null == nil)
    {
      null = [NSNull new]; 
    }
  if (sel != 0) // 如果sel存在,則需要獲取參數(shù)類型,如果sel不存在,則類型是傳過來的
    {
      sig = [self methodSignatureForSelector: sel]; // 獲得方法簽名
      if ([sig numberOfArguments] != 3) // 參數(shù)必須是3個(gè)
    {
      [NSException raise: NSInvalidArgumentException
              format: @"key-value set method has wrong number of args"];
    }
      type = [sig getArgumentTypeAtIndex: 2]; // 第三參數(shù)的type
    }
  if (type == NULL)// 沒有找到對應(yīng)的賦值方式
    {
      [self setValue: val forUndefinedKey:
    [NSString stringWithUTF8String: key]];
    }
  else if ((val == nil || val == null) && *type != _C_ID && *type != _C_CLASS) // 如果obj 等于 null
    {
      [self setNilValueForKey: [NSString stringWithUTF8String: key]];
    }
  else
    {
      switch (*type)
    {
      case _C_ID:
      case _C_CLASS: // 調(diào)用方法,或者直接復(fù)制
        {
          id    v = val;

          if (sel == 0)
        {
          id *ptr = (id *)((char *)self + offset);

          ASSIGN(*ptr, v); // retain v
        }
          else
        {
          void  (*imp)(id, SEL, id) =
            (void (*)(id, SEL, id))[self methodForSelector: sel];

          (*imp)(self, sel, val);
        }
        }
        break;

      case _C_CHR:
        {
          char  v = [val charValue];

          if (sel == 0)
        {
          char *ptr = (char *)((char *)self + offset);

          *ptr = v;
        }
          else
        {
          void  (*imp)(id, SEL, char) =
            (void (*)(id, SEL, char))[self methodForSelector: sel];

          (*imp)(self, sel, v);
        }
        }
        break;
···
// 中間省略的代碼基本上都是根據(jù)類型直接給成員變量賦值或調(diào)用方法
···
      default:
            [self setValue: val forUndefinedKey:
          [NSString stringWithUTF8String: key]];
    }
    }
}

setValue:forKeyPath:

- (void) setValue: (id)anObject forKeyPath: (NSString*)aKey
{
  NSRange       r = [aKey rangeOfString: @"." options: NSLiteralSearch]; // 尋找 . 
#ifdef WANT_DEPRECATED_KVC_COMPAT
  IMP           o = [self methodForSelector: @selector(takeValue:forKeyPath:)];

  setupCompat();
  if (o != takePath && o != takePathKVO)
    {
      (*o)(self, @selector(takeValue:forKeyPath:), anObject, aKey);
      return;
    }
#endif

  if (r.length == 0) // 如果沒有找到 . 則直接調(diào)用 setValue:forKey:
    {
      [self setValue: anObject forKey: aKey];
    }
  else
    {
      NSString  *key = [aKey substringToIndex: r.location];
      NSString  *path = [aKey substringFromIndex: NSMaxRange(r)];
      
      [[self valueForKey: key] setValue: anObject forKeyPath: path]; // 遞歸調(diào)用,直到 沒有 `.`
    }
}

takeValue:forKeyPath:

- (void) takeValue: (id)anObject forKeyPath: (NSString*)aKey
{
  NSRange   r = [aKey rangeOfString: @"." options: NSLiteralSearch];

  GSOnceMLog(@"This method is deprecated, use -setValue:forKeyPath:");
  if (r.length == 0)
    {
      [self takeValue: anObject forKey: aKey];
    }
  else
    {
      NSString  *key = [aKey substringToIndex: r.location];
      NSString  *path = [aKey substringFromIndex: NSMaxRange(r)];

      [[self valueForKey: key] takeValue: anObject forKeyPath: path];
    }
}

可以發(fā)現(xiàn),也是遞歸調(diào)用,直到?jīng)]有 .

valueForKey:

- (id) valueForKey: (NSString*)aKey
{
  unsigned  size = [aKey length] * 8;
  char      key[size + 1];

  [aKey getCString: key
     maxLength: size + 1
      encoding: NSUTF8StringEncoding]; // 轉(zhuǎn)成 char * 字符串
  size = strlen(key);
  return ValueForKey(self, key, size);
}

ValueForKey

static id ValueForKey(NSObject *self, const char *key, unsigned size)
{
  SEL       sel = 0;
  int       off = 0;
  const char    *type = NULL;

  if (size > 0)
    {
      const char    *name;
      char      buf[size + 5];
      char      lo;
      char      hi;

      strncpy(buf, "_get", 4);
      strncpy(&buf[4], key, size);
      buf[size + 4] = '\0';
      lo = buf[4];
      hi = islower(lo) ? toupper(lo) : lo;
      buf[4] = hi;

      name = &buf[1];   // getKey
      sel = sel_getUid(name);
      if (sel == 0 || [self respondsToSelector: sel] == NO)
    {
      buf[4] = lo;
      name = &buf[4];   // key
      sel = sel_getUid(name);
      if (sel == 0 || [self respondsToSelector: sel] == NO)
        {
              buf[4] = hi;
              buf[3] = 's';
              buf[2] = 'i';
              name = &buf[2];   // isKey
              sel = sel_getUid(name);
              if (sel == 0 || [self respondsToSelector: sel] == NO)
                {
                  sel = 0;
                }
        }
    }

      if (sel == 0 && [[self class] accessInstanceVariablesDirectly] == YES)
    {
      buf[4] = hi;
      name = buf;   // _getKey
      sel = sel_getUid(name);
      if (sel == 0 || [self respondsToSelector: sel] == NO)
        {
          buf[4] = lo;
          buf[3] = '_';
          name = &buf[3];   // _key
          sel = sel_getUid(name);
          if (sel == 0 || [self respondsToSelector: sel] == NO)
        {
          sel = 0;
        }
        }
      if (sel == 0)
        {
          if (GSObjCFindVariable(self, name, &type, &size, &off) == NO) // 尋找成員變量 先找 _key
        {
                  buf[4] = hi;
                  buf[3] = 's';
                  buf[2] = 'i';
                  buf[1] = '_';
                  name = &buf[1];   // _isKey
          if (!GSObjCFindVariable(self, name, &type, &size, &off))
                    {
                       buf[4] = lo;
                       name = &buf[4];      // key
               if (!GSObjCFindVariable(self, name, &type, &size, &off))
                         {
                            buf[4] = hi;
                            buf[3] = 's';
                            buf[2] = 'i';
                            name = &buf[2]; // isKey
                            GSObjCFindVariable(self, name, &type, &size, &off);
                         }
                    }
        }
        }
    }
    }
  return GSObjCGetVal(self, key, sel, type, size, off);
}

GSObjCGetVal

id
GSObjCGetVal(NSObject *self, const char *key, SEL sel,
           const char *type, unsigned size, int offset)
{
  NSMethodSignature *sig = nil;

  if (sel != 0)
    {
      sig = [self methodSignatureForSelector: sel]; // 方法簽名
      if ([sig numberOfArguments] != 2) // 參數(shù)必須是兩個(gè)
    {
      [NSException raise: NSInvalidArgumentException
              format: @"key-value get method has wrong number of args"];
    }
      type = [sig methodReturnType]; // 返回值的類型
    }
  if (type == NULL) // 沒有找到方法或者成員變量
    {
      return [self valueForUndefinedKey: [NSString stringWithUTF8String: key]];
    }
  else
    {
      id    val = nil;

      switch (*type) // 根據(jù)返回值類型,調(diào)用方法或者直接訪問成員變量
    {
      case _C_ID:
      case _C_CLASS:
        {
          id    v;

          if (sel == 0)
        {
          v = *(id *)((char *)self + offset);
        }
          else
        {
          id    (*imp)(id, SEL) =
            (id (*)(id, SEL))[self methodForSelector: sel];

          v = (*imp)(self, sel);
        }
          val = v;
        }
        break;

      case _C_CHR:
        {
          signed char   v;

          if (sel == 0)
        {
          v = *(char *)((char *)self + offset);
        }
          else
        {
          signed char   (*imp)(id, SEL) =
            (signed char (*)(id, SEL))[self methodForSelector: sel];

          v = (*imp)(self, sel);
        }
          val = [NSNumber numberWithChar: v];
        }
        break;

      default:
#ifdef __GNUSTEP_RUNTIME__
        {
          Class     cls;
          struct objc_slot  *type_slot;
          SEL       typed;
          struct objc_slot  *slot;

          cls = [self class];
          type_slot = objc_get_slot(cls, @selector(retain));
          typed = GSSelectorFromNameAndTypes(sel_getName(sel), NULL);
          slot = objc_get_slot(cls, typed);
          if (strcmp(slot->types, type_slot->types) == 0)
        {
          return slot->method(self, typed);
        }
        }
#endif
        val = [self valueForUndefinedKey:
          [NSString stringWithUTF8String: key]];
    }
      return val;
    }
}

總結(jié)

以上就是GUNStep關(guān)于KVC中幾個(gè)關(guān)鍵方法的實(shí)現(xiàn).

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌栋烤,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,657評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異屯换,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)与学,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門彤悔,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人索守,你說我怎么就攤上這事晕窑。” “怎么了卵佛?”我有些...
    開封第一講書人閱讀 164,057評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵杨赤,是天一觀的道長蓝丙。 經(jīng)常有香客問我,道長望拖,這世上最難降的妖魔是什么渺尘? 我笑而不...
    開封第一講書人閱讀 58,509評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮说敏,結(jié)果婚禮上鸥跟,老公的妹妹穿的比我還像新娘。我一直安慰自己盔沫,他們只是感情好医咨,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,562評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著架诞,像睡著了一般拟淮。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上谴忧,一...
    開封第一講書人閱讀 51,443評(píng)論 1 302
  • 那天很泊,我揣著相機(jī)與錄音,去河邊找鬼沾谓。 笑死委造,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的均驶。 我是一名探鬼主播昏兆,決...
    沈念sama閱讀 40,251評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼妇穴!你這毒婦竟也來了爬虱?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,129評(píng)論 0 276
  • 序言:老撾萬榮一對情侶失蹤腾它,失蹤者是張志新(化名)和其女友劉穎跑筝,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體携狭,經(jīng)...
    沈念sama閱讀 45,561評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡继蜡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,779評(píng)論 3 335
  • 正文 我和宋清朗相戀三年回俐,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了逛腿。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,902評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡仅颇,死狀恐怖单默,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情忘瓦,我是刑警寧澤搁廓,帶...
    沈念sama閱讀 35,621評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響境蜕,放射性物質(zhì)發(fā)生泄漏蝙场。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,220評(píng)論 3 328
  • 文/蒙蒙 一粱年、第九天 我趴在偏房一處隱蔽的房頂上張望售滤。 院中可真熱鬧,春花似錦台诗、人聲如沸完箩。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽弊知。三九已至,卻和暖如春粱快,著一層夾襖步出監(jiān)牢的瞬間秩彤,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評(píng)論 1 269
  • 我被黑心中介騙來泰國打工事哭, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留呐舔,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,025評(píng)論 2 370
  • 正文 我出身青樓慷蠕,卻偏偏與公主長得像珊拼,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子流炕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,843評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • KVC KVC定義 KVC(Key-value coding)鍵值編碼澎现,就是指iOS的開發(fā)中,可以允許開發(fā)者通過K...
    暮年古稀ZC閱讀 2,142評(píng)論 2 9
  • KVC定義 KVC(Key-value coding)鍵值編碼每辟,就是指iOS的開發(fā)中剑辫,可以允許開發(fā)者通過Key名直...
    SheIsMySin_72e7閱讀 379評(píng)論 0 0
  • KVC(Key-value coding)鍵值編碼,單看這個(gè)名字可能不太好理解渠欺。其實(shí)翻譯一下就很簡單了妹蔽,就是指iO...
    朽木自雕也閱讀 1,560評(píng)論 6 1
  • GNUstep KVC/KVO探索(一):KVC的內(nèi)部實(shí)現(xiàn)GNUstep KVC/KVO探索(二):KVO的內(nèi)部實(shí)...
    哦呵呵y閱讀 1,090評(píng)論 0 2
  • 上篇文章講完了KVO這篇來看看KVC,KVC在開發(fā)中的使用也算是挺多的挠将,下面從幾個(gè)方面來看看KVC 常見的API用...
    木子雨廷t閱讀 433評(píng)論 0 5