iOS-底层探索18:KVC 底层原理

iOS 底层探索 文章汇总

目录



相关面试题:谈谈你对KVC的理解

KVC的全称是Key-Value Coding,翻译成中文是键值编码,键值编码是由NSKeyValueCoding非正式协议启用的一种机制,对象采用该协议来间接访问其属性。即可以通过一个字符串key来访问某个属性。这种间接访问机制补充了实例变量及其相关的访问器所提供的直接访问。

一、KVC基本使用

    // 1:基本类型使用
    NAPerson *person = [NAPerson new];
    [person setValue:@"differ" forKey:@"name"];
    CJLog(@"%@",person.name);
    
    // 2:集合类型使用
    person.array = @[@"1",@"2",@"3"];
    //person.array[0] = @"100"; 不能直接修改数组中的元素
    // 方式一:创建一个新的数组
    NSArray *array = @[@"100",@"2",@"3"];
    [person setValue:array forKey:@"array"];
    CJLog(@"%@",[person valueForKey:@"array"]);
    
    // 方式二:构建可变数组
    NSMutableArray *mArray = [person mutableArrayValueForKey:@"array"];
    mArray[0] = @"200";
    CJLog(@"%@",[person valueForKey:@"array"]);
    
    // 3:集合操作符使用
    // 遍历
    NSEnumerator *enumerator = [array objectEnumerator];
    NSString *str = nil;
    while (str = [enumerator nextObject]) {
        CJLog(@"%@", str);
    }
    
    // 字典操作
    NSDictionary *dict = @{@"name":@"differ",@"subject":@"iOS",};
    NAStudent *p = [[NAStudent alloc] init];
    // 字典转模型
    [p setValuesForKeysWithDictionary:dict];
    // 键数组转模型到字典
    NSArray *keyArray = @[@"name",@"subject"];
    NSDictionary *dic = [p dictionaryWithValuesForKeys:keyArray];

    // 聚合操作符 @avg、@count、@max、@min、@sum
    // 数组操作符 @distinctUnionOfObjects @unionOfObjects
    // 嵌套集合(array&set)操作 @distinctUnionOfArrays @unionOfArrays @distinctUnionOfSets
    
    // 4:访问非对象属性 - 面试可能问到
    // 结构体-官方文档示例
    ThreeFloats floats = {1.,2.,3.};
    NSValue *value     = [NSValue valueWithBytes:&floats objCType:@encode(ThreeFloats)];
    [person setValue:value forKey:@"threeFloats"];
    NSValue *value1    = [person valueForKey:@"threeFloats"];
    CJLog(@"%@",value1);
    
    ThreeFloats th;
    [value1 getValue:&th];
    CJLog(@"%f-%f-%f",th.x,th.y,th.z);
    
    // 5:KVC - 层层访问 - keyPath
    NAStudent *student = [NAStudent alloc];
    student.subject    = @"differ";
    person.student     = student;
    
    [person setValue:@"Swift" forKeyPath:@"student.subject"];
    CJLog(@"%@",[person valueForKeyPath:@"student.subject"]);

typedef struct {
    float x, y, z;
} ThreeFloats;

进入setValue:forKey:方法我们可以看到该方法在NSObjectNSKeyValueCoding分类中,因此所有继承于NSObject的类均有KVC功能。



官方文档地址:Key-Value Coding Programming Guide


二、KVC设值、取值底层分析

在官方文档Search Pattern for the Basic Setter描述中可以看到KVC的设值过程。

Search Pattern for the Basic Setter

The default implementation of setValue:forKey:, given key and value parameters as input, attempts to set a property named key to value (or, for non-object properties, the unwrapped version of value, as described in Representing Non-Object Values) inside the object receiving the call, using the following procedure:

  1. Look for the first accessor named set<Key>: or _set<Key>, in that order. If found, invoke it with the input value (or unwrapped value, as needed) and finish.

  2. If no simple accessor is found, and if the class method accessInstanceVariablesDirectly returns YES, look for an instance variable with a name like _<key>, _is<Key>, <key>, or is<Key>, in that order. If found, set the variable directly with the input value (or unwrapped value) and finish.

  3. Upon finding no accessor or instance variable, invoke setValue:forUndefinedKey:. This raises an exception by default, but a subclass of NSObject may provide key-specific behavior.

根据官方文档可以绘制如下设值流程图:

  • 设值方法的顺序为:setName->_setName->setIsName
  • 寻找变量的顺序为:_name ->_isName ->name->isName(没有定义前面的成员变量才会向后寻找)

取值方法:[person valueForKey@"name"];

  • 取值方法的顺序为:getName->name->isName -> _name
  • 寻找变量的顺序为:_name ->_isName ->name->isName

三、简单自定义KVC设值、取值

这里通过简单自定义KVC设值、取值加深对KVC底层机制的理解

自定义KVC设置流程,主要分为以下几个步骤:
  1. 判断key非空
  2. 查找setter方法,顺序是:setKey、_setKey、 setIsKey
  3. 判断accessInstanceVariablesDirectly方法的返回值,即是否允许间接访问成员变量
    3.1 返回YES,继续下一步设值,
    3.2 返回NO,抛出异常
  4. 间接访问变量赋值(只会走一次),顺序是:_key、_isKey、key、isKey
    4.1 定义一个收集实例变量的数组
    4.2 通过class_getInstanceVariable方法,获取相应的ivar
    4.3 通过object_setIvar方法,对相应的ivar设置值
  5. 如果找不到相关实例变量,则抛出异常
//设值
- (void)lcj_setValue:(nullable id)value forKey:(NSString *)key{
    
//    1、判断key 是否存在
    if (key == nil || key.length == 0) return;
    
//    2、找setter方法,顺序是:setKey、_setKey、 setIsKey
    // key 要大写
    NSString *Key = key.capitalizedString;
    // key 要大写
    NSString *setKey   = [NSString stringWithFormat:@"set%@:", Key];
    NSString *_setKey  = [NSString stringWithFormat:@"_set%@:", Key];
    NSString *setIsKey = [NSString stringWithFormat:@"setIs%@:", Key];
    
    if ([self lcj_performSelectorWithMethodName:setKey value:value]) {
        NSLog(@"*************%@*************", setKey);
        return;
    } else if([self lcj_performSelectorWithMethodName:_setKey value:value]) {
        NSLog(@"*************%@*************", _setKey);
        return;
    } else if([self lcj_performSelectorWithMethodName:setIsKey value:value]) {
        NSLog(@"*************%@*************", setIsKey);
        return;
    }
    
    
//    3、判断accessInstanceVariablesDirectly方法的返回值,即是否允许间接访问成员变量,返回YES,继续下一步设值,如果是NO,则抛出异常
    if (![self.class accessInstanceVariablesDirectly]) {
        @throw [NSException exceptionWithName:@"CJLUnKnownKeyException" reason:[NSString stringWithFormat:@"****[%@ valueForUndefinedKey:]: this class is not key value coding-compliant for the key name.****",self] userInfo:nil];
    }
    
//    4、间接访问变量赋值,顺序为:_key、_isKey、key、isKey
    // 4.1 定义一个收集实例变量的数组
    NSMutableArray *mArray = [self getIvarListName];
    // _<key> _is<Key> <key> is<Key>
    NSString *_key   = [NSString stringWithFormat:@"_%@", key];
    NSString *_isKey = [NSString stringWithFormat:@"_is%@", key];
    NSString *isKey  = [NSString stringWithFormat:@"is%@", key];
    if ([mArray containsObject:_key]) {
        // 4.2 获取相应的 ivar
        Ivar ivar = class_getInstanceVariable([self class], _key.UTF8String);
        // 4.3 对相应的 ivar 设置值
        object_setIvar(self, ivar, value);
        return;
    } else if ([mArray containsObject:_isKey]) {
        Ivar ivar = class_getInstanceVariable([self class], _isKey.UTF8String);
        object_setIvar(self, ivar, value);
        return;
    } else if ([mArray containsObject:key]) {
        Ivar ivar = class_getInstanceVariable([self class], key.UTF8String);
        object_setIvar(self, ivar, value);
        return;
    } else if ([mArray containsObject:isKey]) {
        Ivar ivar = class_getInstanceVariable([self class], isKey.UTF8String);
        object_setIvar(self, ivar, value);
        return;
    }
    
//    5、如果找不到则抛出异常
    @throw [NSException exceptionWithName:@"LCJUnknownKeyException" reason:[NSString stringWithFormat:@"****[%@ %@]: this class is not key value coding-compliant for the key name.****",self,NSStringFromSelector(_cmd)] userInfo:nil];
}


自定义KVC取置流程,主要分为以下几个步骤:
  1. 判断key非空
  2. 查找相应方法,顺序是:get<Key>、<key>、is<Key>、 _<key>(NSArray还会寻找countOf<Key> and objectIn<Key>AtIndex)
  3. 判断accessInstanceVariablesDirectly方法的返回值,即是否允许间接访问成员变量
    3.1 返回YES,继续下一步设值,
    3.2 返回NO,抛出异常
  4. 间接访问实例变量,顺序依然是:_<key>、 _is<Key>、 <key>、 is<Key>
    4.1 定义一个收集实例变量的数组
    4.2 通过class_getInstanceVariable方法,获取相应的ivar
    4.3 通过object_getIvar方法,返回相应的 ivar的值
  5. 如果找不到相关实例变量,返回空字符串
//取值
- (nullable id)lcj_valueForKey:(NSString *)key {
    
//    1、判断非空
    if (key == nil || key.length == 0) {
        return nil;
    }
    
//    2、找到相关方法:get<Key>、 <key> 、is<Key>、 _<key>、countOf<Key> 、objectIn<Key>AtIndex
    // key 要大写
    NSString *Key = key.capitalizedString;
    // 拼接方法
    NSString *getKey = [NSString stringWithFormat:@"get%@",Key];
    NSString *countOfKey = [NSString stringWithFormat:@"countOf%@",Key];
    NSString *objectInKeyAtIndex = [NSString stringWithFormat:@"objectIn%@AtIndex:",Key];

#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
    if ([self respondsToSelector:NSSelectorFromString(getKey)]) {
        return [self performSelector:NSSelectorFromString(getKey)];
    } else if ([self respondsToSelector:NSSelectorFromString(key)]) {
        return [self performSelector:NSSelectorFromString(key)];
    } else if (is<Key>、 _<key>) {}
    //集合类型
    else if ([self respondsToSelector:NSSelectorFromString(countOfKey)]) {
        if ([self respondsToSelector:NSSelectorFromString(objectInKeyAtIndex)]) {
            int num = (int)[self performSelector:NSSelectorFromString(countOfKey)];
            NSMutableArray *mArray = [NSMutableArray arrayWithCapacity:1];
            for (int i = 0; i<num-1; i++) {
                num = (int)[self performSelector:NSSelectorFromString(countOfKey)];
            }
            for (int j = 0; j<num; j++) {
                id objc = [self performSelector:NSSelectorFromString(objectInKeyAtIndex) withObject:@(num)];
                [mArray addObject:objc];
            }
            return mArray;
        }
    }

#pragma clang diagnostic pop
    
//    3、判断accessInstanceVariablesDirectly方法的返回值,即是否允许间接访问成员变量,返回YES,继续下一步设值,如果是NO,则抛出异常
    if (![self.class accessInstanceVariablesDirectly]) {
        @throw [NSException exceptionWithName:@"CJLUnKnownKeyException" reason:[NSString stringWithFormat:@"****[%@ valueForUndefinedKey:]: this class is not key value coding-compliant for the key name.****",self] userInfo:nil];
    }
    
//    4.找相关实例变量进行赋值,顺序为:_<key>、 _is<Key>、 <key>、 is<Key>
    // 4.1 定义一个收集实例变量的数组
    NSMutableArray *mArray = [self getIvarListName];
    // 例如:_name -> _isName -> name -> isName
    NSString *_key   = [NSString stringWithFormat:@"_%@",key];
    NSString *_isKey = [NSString stringWithFormat:@"_is%@",Key];
    NSString *isKey  = [NSString stringWithFormat:@"is%@",Key];
    if ([mArray containsObject:_key]) {
        Ivar ivar = class_getInstanceVariable([self class], _key.UTF8String);
        return object_getIvar(self, ivar);
    } else if ([mArray containsObject:_isKey]) {
        Ivar ivar = class_getInstanceVariable([self class], _isKey.UTF8String);
        return object_getIvar(self, ivar);
    } else if ([mArray containsObject:key]) {
        Ivar ivar = class_getInstanceVariable([self class], key.UTF8String);
        return object_getIvar(self, ivar);
    } else if ([mArray containsObject:isKey]) {
        Ivar ivar = class_getInstanceVariable([self class], isKey.UTF8String);
        return object_getIvar(self, ivar);
    }

    return @"";
}


四、KVC 使用场景

1、动态设值和取值
  • 常用的可以通过setValue:forKey:valueForKey:
  • 也可以通过路由的方式setValue:forKeyPath:valueForKeyPath:
2、通过KVC访问和修改私有变量

在日常开发中,对于类的私有属性,在外部定义的对象,是无法直接访问私有属性的,但是对于KVC而言,一个对象没有自己的隐私,所以可以通过KVC修改和访问任何私有属性

3、多值操作(model和字典互转)

model和字典的转换可以通过下面两个KVCAPI实现:

//字典转模型
- (void)setValuesForKeysWithDictionary:(NSDictionary<NSString *, id> *)keyedValues;

//模型转字典
- (NSDictionary<NSString *, id> *)dictionaryWithValuesForKeys:(NSArray<NSString *> *)keys;
4、修改一些系统空间的内部属性

在日常开发中,我们知道,很多UI控件都是在其内部由多个UI空间组合而成,这些内部控件苹果并没有提供访问的API,但是使用KVC可以解决这个问题,常用的就是自定义tabbar、个性化UITextField中的placeHolderTextiOS13开始逐步不允许通过valueForKey、setValue: forKey获取和设置私有属性,如:_placeholderLabel.textColor)。

5、用KVC实现高阶消息传递

在对容器类使用KVC时,valueForKey:将会被传递给容器中的每一个对象,而不是对容器本身进行操作,结果会被添加到返回的容器中,这样可以很方便的操作集合 来返回 另一个集合

//KVC实现高阶消息传递
- (void)transmitMsg{
    NSArray *arrStr = @[@"english", @"franch", @"chinese"];
    NSArray *arrCapStr = [arrStr valueForKey:@"capitalizedString"];
    
    for (NSString *str in arrCapStr) {
        CJLog(@"%@", str);
    }
    
    NSArray *arrCapStrLength = [arrCapStr valueForKeyPath:@"capitalizedString.length"];
    for (NSNumber *length in arrCapStrLength) {
        CJLog(@"%ld", (long)length.integerValue);
    }
}

// 打印结果:
English
Franch
Chinese
7
6
7


参考:

DIS_KVC_KVO自定义实现

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容