OC-KVC的本质及原理

面试题 - KVC

通过KVC修改属性会触发KVO么?

KVC的赋值和取值过程是怎样的?原理是什么?

KVC的基本使用

image-20210417151743134

KVC的全称是Key-Value Coding,俗称“键值编码”,可以通过一个key来访问某个属性

常见的API有:

- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
- (void)setValue:(id)value forKey:(NSString *)key;
- (id)valueForKeyPath:(NSString *)keyPath;
- (id)valueForKey:(NSString *)key;

KVC的基本使用:

@interface MJCat : NSObject
@property (assign, nonatomic) int weight;
@end

@interface MJPerson : NSObject
@property (assign, nonatomic) int age;

@property (retain, nonatomic) MJCat *cat;
@end
@implementation MJPerson
@end
@implementation MJCat
@end
      person.age = 10;
        
        NSLog(@"%@", [person valueForKey:@"age"]);
        NSLog(@"%@", [person valueForKeyPath:@"cat.weight"]);
        
        
        NSLog(@"%d", person.age);
        
        [person setValue:[NSNumber numberWithInt:10] forKey:@"age"];
        [person setValue:@30 forKey:@"age"];
        person.cat = [[MJCat alloc] init];
        [person setValue:@20 forKeyPath:@"cat.weight"];
        
        NSLog(@"%d", person.age);
        NSLog(@"%d", person.cat.weight);

RUN>

=================打印结果=================
2021-04-17 15:29:31.379302+0800 Interview01-KVC[2187:88935] 10
2021-04-17 15:29:31.379738+0800 Interview01-KVC[2187:88935] (null)
2021-04-17 15:29:31.379774+0800 Interview01-KVC[2187:88935] 10
2021-04-17 15:29:31.380053+0800 Interview01-KVC[2187:88935] 30
2021-04-17 15:29:31.380093+0800 Interview01-KVC[2187:88935] 20

通过KVC给属性赋值能触发KVO吗?

能不能试一下就知道了:

我们创建一个MJPerson类,添加一个age属性,然后给这个age属性添加一个KVO,使用KVC修改age的值:

#import "MJObserver.h"

@implementation MJObserver

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
{
    NSLog(@"observeValueForKeyPath - %@", change);
}

@end
MJObserver *observer = [[MJObserver alloc] init];
MJPerson *person = [[MJPerson alloc] init];
// 添加KVO监听
[person addObserver:observer forKeyPath:@"age" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL];
[person setValue:@10 forKey:@"age"];

RUN>

=================打印结果=================
2021-04-17 15:34:55.609944+0800 Interview01-KVC[2235:92111] observeValueForKeyPath - {
    kind = 1;
    new = 10;
    old = 0;
}

可以看到KVC访问属性的时候,的确调用了setAge:方法.那么setVlaue:forKey:是如何设值的呢?

通过KVC修改age属性会触发KVO。为什么呢?先往下看

接下来看看setValue:forKey:设值原理和valueForKey:取值原理。

KVC设值原理

KVC赋值流程

执行步骤:
1:setVlaue:forKey:首先查找setKey:方法,如果有就执行;如果没有再去查找_setKey:方法.
2:如果没有找到_setKey:方法,就去查看accessInstanceVariablesDirectly方法的返回值,这个方法的意思是:是否允许直接进入对象的成员变量?,如果返回YES,就按照_key,_isKey,key,isKey的顺序查找成员变量,如果找到直接赋值;如果没找到就抛出异常NSUnknownKeyException.
3:如果accessInstanceVariablesDirectly方法直接返回NO(默认返回YES),就直接抛出异常NSUnknownKeyException.

设值原理解释:

1.寻找setAge方法
- (void)setAge:(int)age
{
    NSLog(@"setAge: - %d", age);
}

2.寻找_setAge方法
- (void)_setAge:(int)age
{
    NSLog(@"_setAge: - %d", age);
}

3.找不到上面两个方法就调用accessInstanceVariablesDirectly问问能不能直接访问成员变量
 默认的返回值就是YES
+ (BOOL)accessInstanceVariablesDirectly
{
    return YES;
}

4.1 如果返回NO,就调用setValue:forUndefinedKey:并抛出异常NSUnknownKeyException
4.2如果返回YES,会按顺序_key,_isKey,key,isKey赋值,如果四个都找不到就报上面的错

上面我们知道KVC设值会触发KVO,但是如果没有set方法呢?就算没有set方法只有成员变量,通过KVC进行赋值也会触发KVO,可以理解它们是配套使用的。

我们先证明第一点(执行步骤1),把MJPerson类中属性删掉,然后重写setAge:,_setAge:方法:

@implementation MJPerson

- (void)setAge:(int)age{
    NSLog(@"setAge方法被调用");
}

- (void)_setAge:(int)age{
    NSLog(@"_setAge被调用");
}

@end
MJPerson *person = [[MJPerson alloc] init];
// 通过KVC修改age属性
[person setValue:@10 forKey:@"age"];

RUN>

=================打印结果=================
2021-04-17 15:42:35.598146+0800 Interview01-KVC[2348:98038] setAge方法被调用

我们把setAge:方法注释掉,再运行:

=================打印结果=================
2021-04-17 15:45:15.169572+0800 Interview01-KVC[2404:100780] _setAge被调用

通过运行结果可以很清楚的看到,[person setValue:@10 forKey:@"age"];会先去查找setAge:方法,如果有就调用,如果没有再去查找_setAge:方法.

接下来验证第二点(执行步骤2),我们在MJPerson注释掉刚才的setAge:方法,然后重写accessInstanceVariablesDirectly方法,然后返回NO,运行下看看效果:

@implementation MJPerson

+ (BOOL)accessInstanceVariablesDirectly{
    return NO;
}

@end
MJPerson *person = [[MJPerson alloc] init];
// 通过KVC修改age属性
[person setValue:@10 forKey:@"age"];

RUN>

=================打印结果=================
2021-04-17 15:48:20.837291+0800 Interview01-KVC[2437:102949] *** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: '[<MJPerson 0x102892cb0> setValue:forUndefinedKey:]: this class is not key value coding-compliant for the key age.'
image-20210417154937044

再修改为return YES;然后添加如下成员变量:

@interface MJPerson : NSObject
{
@public
    int _age;
    int _isAge;
    int age;
    int isAge;
}

@end
@implementation MJPerson

+ (BOOL)accessInstanceVariablesDirectly{
    return YES;
}

@end
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        MJPerson *person = [[MJPerson alloc] init];
        // 通过KVC修改age属性
        [person setValue:@10 forKey:@"age"];              
        NSLog(@"%@", [person valueForKey:@"age"]);
    }
    return 0;
}

RUN>

=================打印结果=================
2021-04-17 16:05:21.917933+0800 Interview01-KVC[2574:111457] 10
image-20210417160635084
image-20210417160710325
image-20210417160805503

有人可能会以为是成员变量顺序导致的,我们把成员变量的顺序打乱在执行看看:

image-20210417160931763

我们把4个属性全注释掉,运行一下看看效果:

@interface MJPerson : NSObject
{
}

RUN>

=================打印结果=================
2021-04-17 16:10:24.189710+0800 Interview01-KVC[2648:117035] *** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: '[<MJPerson 0x10201af80> setValue:forUndefinedKey:]: this class is not key value coding-compliant for the key age.'

如果找到直接赋值;如果没找到就抛出异常NSUnknownKeyException.

如果用KVC修改一个类的成员变量就会触发KVO.我们来验证一下

@interface MJPerson : NSObject
{
@public
    int age;
}

@end
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        MJObserver *observer = [[MJObserver alloc] init];
        MJPerson *person = [[MJPerson alloc] init];
        
        // 添加KVO监听
        [person addObserver:observer forKeyPath:@"age" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL];
        
        // 通过KVC修改age属性
        [person setValue:@10 forKey:@"age"];
    }
    return 0;
}

RUN>

=================打印结果=================
2021-04-17 16:15:52.595089+0800 Interview01-KVC[2690:120312] observeValueForKeyPath - {
    kind = 1;
    new = 10;
    old = 0;
}

其实KVC内部调用了下面方法才会触发KVO的

[person willChangeValueForKey:@"age"];
person->_age = 10;
[person didChangeValueForKey:@"age"];

因为KVC赋值同样会调用willChangeValueForKey:,didChangeValueForKey:两个方法,我们在MJPerson中重写这两个方法:

- (void)willChangeValueForKey:(NSString *)key
{
    [super willChangeValueForKey:key];
    NSLog(@"willChangeValueForKey - %@", key);
}

- (void)didChangeValueForKey:(NSString *)key
{
    NSLog(@"didChangeValueForKey - begin - %@", key);
    [super didChangeValueForKey:key];
    NSLog(@"didChangeValueForKey - end - %@", key);
}

RUN>

=================打印结果=================
2021-04-17 16:18:17.974684+0800 Interview01-KVC[2707:121945] willChangeValueForKey - age
2021-04-17 16:18:17.975060+0800 Interview01-KVC[2707:121945] didChangeValueForKey - begin - age
2021-04-17 16:18:17.975254+0800 Interview01-KVC[2707:121945] observeValueForKeyPath - {
    kind = 1;
    new = 10;
    old = 0;
}
2021-04-17 16:18:17.975309+0800 Interview01-KVC[2707:121945] didChangeValueForKey - end - age

可以看到,这两个方法的确都被调用了,所以KVC给一个类的属性赋值也能触发KVO.

KVC取值原理

取值原理.png

KVC取值原理解释

1.getAge
- (int)getAge
{
    return 11;
}

2.age
- (int)age
{
    return 12;
}

3.isAge
- (int)isAge
{
    return 13;
}

4._age
- (int)_age
{
    return 14;
}

5.3.找不到上面四个方法就调用accessInstanceVariablesDirectly问问能不能直接访问成员变量
 默认的返回值就是YES
+ (BOOL)accessInstanceVariablesDirectly
{
    return YES;
}

5.1 如果返回NO,就调用valueforUndefinedKey:并抛出异常NSUnknownKeyException
5.2如果返回YES,会按顺序_key,_isKey,key,isKey赋值,如果四个都找不到就报上面的错
@implementation MJPerson

- (int)getAge
{
    return 11;
}

- (int)age
{
    return 12;
}

- (int)isAge
{
    return 13;
}

- (int)_age
{
    return 14;
}

@end
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        MJPerson *person = [[MJPerson alloc] init];
        [person setValue:@18 forKey:@"age"];
        
        NSLog(@"%@", [person valueForKey:@"age"]);
    }
    return 0;
}

Run>

2021-04-17 16:21:41.302038+0800 Interview01-KVC[2757:124317] 11
image-20210417162323944
image-20210417162342102

找不到上面四个方法就调用accessInstanceVariablesDirectly

image-20210417162508473
@implementation MJPerson

- (instancetype)init
{
    self = [super init];
    if (self) {
        _age=11;
        _isAge=21;
        age=31;
        isAge = 41;
    }
    return self;
}

// 默认的返回值就是YES
+ (BOOL)accessInstanceVariablesDirectly
{
    return YES;
}

@end
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        MJPerson *person = [[MJPerson alloc] init];
 
        
        NSLog(@"%@", [person valueForKey:@"age"]);
    }
    return 0;
}

RUN>

=================打印结果=================
2021-04-17 16:32:46.330006+0800 Interview01-KVC[2902:132396] 11

面试题 - KVC

通过KVC修改属性会触发KVO么?

会触发KVO.

KVC的赋值和取值过程是怎样的?原理是什么?

看 KVC设值原理 KVC取值原理 步骤,两张图的流程

特别备注

本系列文章总结自MJ老师在腾讯课堂iOS底层原理班(下)/OC对象/关联对象/多线程/内存管理/性能优化,相关图片素材均取自课程中的课件。如有侵权,请联系我删除,谢谢!

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

推荐阅读更多精彩内容

  • 夜莺2517阅读 127,718评论 1 9
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,535评论 28 53
  • 兔子虽然是枚小硕 但学校的硕士四人寝不够 就被分到了博士楼里 两人一间 在学校的最西边 靠山 兔子的室友身体不好 ...
    待业的兔子阅读 2,601评论 2 9
  • 信任包括信任自己和信任他人 很多时候,很多事情,失败、遗憾、错过,源于不自信,不信任他人 觉得自己做不成,别人做不...
    吴氵晃阅读 6,187评论 4 8