首先我们了解一下kvo和kvc的名次解释
KVO的全称是Key-Value Observing,俗称“键值监听”,可以用于监听某个对象属性值的改变
这里给出几道kvo经典的面试题
iOS用什么方式实现对一个对象的KVO?(KVO的本质是什么?)
如何手动触发KVO?
直接修改成员变量会触发KVO么?
KVC的赋值和取值过程是怎样的?原理是什么?
通过KVC修改属性会触发KVO么?
在接下来讲解的过程中会一一解答
//这里给出了我们kvo最基本的使用方法
self.person1 = [[CCPerson alloc] init];
self.person1.age = 1;
// 给person1对象添加KVO监听
NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
[self.person1 addObserver:self forKeyPath:@"age" options:options context:@"123"];
// 当监听对象的属性值发生改变时,就会调用
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
{
NSLog(@"监听到%@的%@属性值改变了 - %@ - %@", object, keyPath, change, context);
}
通过runtimeAPI我们可以知道使用了KVO监听的对象isa指针会指向另一个类NSKVONotifying_CCPerson,同时这个类的内部会调用几个方法。
从图中我们不难发现
1.Person对象利用RuntimeAPI动态生成一个子类,并且让instance对象的isa指向这个全新的子类
2.当修改instance对象的属性时,会调用Foundation的_NSSetXXXValueAndNotify函数
3.willChangeValueForKey:
4.父类原来的setter
5.didChangeValueForKey:
6.内部会触发监听器(Oberser)的监听方法( observeValueForKeyPath:ofObject:change:context:)
查看_NSSet*AndNotify的存在
_NSSet*ValueAndNotify的内部实现
调用willChangeValueForKey:
调用原来的setter实现
调用didChangeValueForKey:
didChangeValueForKey:内部会调用observer的observeValueForKeyPath:ofObject:change:context:方法
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;