【iOS面试粮食】OC语言—KVC、KVO

本文章将记录有关 KVC、KVO的特性,如有错误欢迎指出~

KVC(Key-Value Coding)键值编码

基于Object-C的语言特性,KVC可以让我们在开发中直接通过对象的字符串参数(Key)获取、赋值对象的属性。那我们就可以通过KVC的特性来修改控件的私有属性,是不是很刺激~

KVC的操作方法由NSKeyValueCoding协议提供,而NSObject就实现了这个协议,也就是说Object-C中几乎所有的对象都支持KVC操作,常用的KVC操作方法如下:

- (nullable id)valueForKey:(NSString *)key;                          //直接通过属性名来取值

- (void)setValue:(nullable id)value forKey:(NSString *)key;          //通过属性名来设值

- (nullable id)valueForKeyPath:(NSString *)keyPath;                  //通过属性路径来取值

- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath;  //通过属性路径来设值

接下来,我们看下通过属性名设置值方法setValue:forKey:的流程图

setValue:forKey:方法设置值流程图

当我们调用setValue:forKey:设置属性时

  • 优先调用setKey、_setKey方法
  • 没找到上面的方法, 则调用accessInstanceVariablesDirectly方法,该方法默认返回YES,会按照_key,_iskey,key,iskey的顺序查找成员变量
  • 如果按照上面的顺序都搜索不到成员变量,则会调用setValue:forUndefinedKey:,并抛出异常

注意查找过程中不管这些方法、成员变量是私有的还是公共的都能正确设置

了解完设置属性,再来看看valueForkey:方法取值的流程图

valueForkey: 方法取值流程图

可以看到,整个流程和设置属性值的步骤是一模一样的,只不过查找的方法不一样,取值的时候

  • 优先按照getKey、key、isKey、_key 的顺序查找方法
  • 找不到上面的方法,会按照_key,_iskey,key,iskey的顺序查找成员变量
  • 如果按照上面的顺序都搜索不到成员变量,则会调用setValue:forUndefinedKey:,并抛出异常

注意查找过程中不管这些方法、成员变量是私有的还是公共的都能正确读取到值

KVO(Key-Value Observing)键值观察

KVO是一种观察者模式的衍生,用于监听某个对象属性值的改变。

简单来说KVO可以通过监听对象属性的key,来获得value的变化,利用它可以在对象之间监听值的变化。

Objective-C中要实现KVO则必须实现NSKeyValueObServing协议,而NSObject已经实现了改协议,因此对于所有继承了NSObject的类型,也就是说Object-C中几乎所有的对象都支持KVO操作,常用的KVO操作方法如下:

/**
注册观察者
observer:观察者,也就是KVO通知的订阅者。
keyPath:描述将要观察的属性,相当于被观察者。
options:KVO的一些属性配置。
        NSKeyValueObservingOptionNew:change字典包括改变后的值
        NSKeyValueObservingOptionOld:change字典包括改变前的值
        NSKeyValueObservingOptionInitial:注册后立刻触发KVO通知
        NSKeyValueObservingOptionPrior:值改变前是否也要通知(这个key决定了是否在改变前改变后通知两次)
                 
context: 上下文,这个会传递到订阅着的函数中,用来区分消息。
 */
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;

// 移除观察者
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(nullable void *)context 
  
// 移除观察者
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;


// 监听回调方法, change 这个字典保存了变更信息,具体是哪些信息取决于注册观察者时的options
- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary *)change
                       context:(void *)context


使用KVO的整个流程就是

  • 注册观察者 addObserver: forKeyPath: options: context:
  • 实现回调方法 observeValueForKeyPath: ofObject: change: context:
  • 在合适的时机,移除观察者 removeObserver: forKeyPathremoveObserver: forKeyPath: context:

简单的应用代码表现为

KLPerson.h
@interface KLPerson : NSObject
// 公开属性
@property (nonatomic, readwrite, copy) NSString *name;

@end

KLPerson.m
@implementation KLPerson {
  // 私有属性
    int _age;
}

#pragma mark    ---     Lifecycle

- (void)dealloc {
    // 移除观察者
    [self.person removeObserver:self forKeyPath:@"name"];
    [self.person removeObserver:self forKeyPath:@"age"];
}

- (void)viewDidLoad {
    [super viewDidLoad];
 
    // 监听 Person 的公开属性
    [self.person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];
    
     // 监听 Person 的私有属性
    [self.person addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];
    
    self.person.name = @"小红";
  
    // 使用KVC对私有属性赋值
    [self.person setValue:@10 forKey:@"age"];

    
}

#pragma mark    ---     OverwriteSuperClass
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context{
    
    // 通常为以下的写法
    if (object == self.person && [keyPath isEqualToString:@"name"]) {
        // 做些什么...
        
    } else if (object == self.person && [keyPath isEqualToString:@"age"]) {
        // 做些什么...
        
    } else {
        
        [super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
    }
    
     NSLog(@"监听到%@的%@改变了%@", object, keyPath,change);
}


输出的结果为:
  监听到<KLPerson: 0x600003008560>的name改变了{
    kind = 1;
    new = "小红";
    old = "<null>";
},
  监听到<KLPerson: 0x600003008560>的age改变了{
    kind = 1;
    new = 10;
    old = 0;
}

从输出的Log,可以看得出来,当被观察者对象的属性值改变时,观察者可以通过 observeValueForKeyPath: ofObject: change: context:回调方法获取到改变的值,去搞些事情~

使用KVC对私有属性赋值时,也会触发回调~

如果想要了解更多 KVO底层原理的实现,可以看下这篇文章,很详细iOS底层原理总结 - 探寻KVO本质

Q & A

  1. Q :iOS用什么方式实现对一个对象的KVO?(KVO的本质是什么?)

  2. A :当一个对象使用了KVO监听,iOS系统会修改这个对象的isa指针,改为指向一个全新的通过Runtime动态创建的子类 NSKVONotifying_对象的类,子类拥有自己的set方法实现,set方法实现内部会顺序调用

    • willChangeValueForKey:方法
    • 原来的setter方法实现
    • didChangeValueForKey:方法

    didChangeValueForKey:方法内部又会调用监听器的observeValueForKeyPath:ofObject:change:context:监听方法。

  3. Q : 如何手动触发KVO?

  4. A :被监听的属性的值被修改时,就会自动触发KVO。如果想要手动触发KVO,则需要我们自己调用willChangeValueForKey:didChangeValueForKey:方法,即可在不改变属性值的情况下手动触发KVO,并且这两个方法缺一不可。

参考资料

iOS底层原理总结 - 探寻KVO本质

iOS开发系列--Objective-C之KVC、KVO

iOS开发技巧系列---详解KVC(我告诉你KVC的一切)

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

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 1.KVC 关于 KVC 和 KVO ,我之前的总结文章有写过,但是趋于表面,没有探究其内部真正的实现原理和进阶用...
    Liberalism阅读 1,080评论 0 5
  • KCV 其实由于ObjC的语言特性,你根部不必进行任何操作就可以进行属性的动态读写,这种方式就是Key Value...
    TYM阅读 1,054评论 0 4
  • 1. KVO 一.KVO原理的使用与证明 我们在开发的过程中经常使用KVO和KVC,但是我们并不了解其底层原理和功...
    周灬阅读 842评论 0 9
  • 今天早上起床后看到孩子的手机在客厅充电,已经有几天都是这样的,心里一下小小的感动,真的是这样,你表达了对孩子的爱...
    李美华_16阅读 211评论 0 3