键路径(keyPath)
在一个给定的实体中,同一个属性的所有值具有相同的数据类型。
键-值编码技术用于进行这样的查找—它是一种间接访问对象属性的机制。
键路径是一个由用点作分隔符的键组成的字符串,用于指定一个连接在一起的对象性质序列。
第一个键的性质是由先前的性质决定的,接下来每个键的值也是相对于其前面的性质。
键路径使您可以以独立于模型实现的方式指定相关对象的性质。
通过键路径,您可以指定对象图中的一个任意深度的路径,使其指向相关对象的特定属性。
键值编码(KVC)
键值编码是一种间接访问对象的属性使用字符串来标识属性,
而不是通过调用存取方法,直接或通过实例变量访问的机制,
非对象类型的变量将被自动封装或者解封成对象,很多情况下会简化程序代码;
KVC的缺点:
一旦使用 KVC 你的编译器无法检查出错误,即不会对设置的键、键路径进行错误检查,
且执行效率要低于合成存取器方法和自定的 setter 和 getter 方法。
因为使用 KVC 键值编码,它必须先解析字符串,然后在设置或者访问对象的实例变量。
键值观察(KVO)
键值观察机制是一种能使得对象获取到其他对象属性变化的通知 ,极大的简化了代码。
实现 KVO 键值观察模式,被观察的对象必须使用 KVC 键值编码来修改它的实例变量,
这样才能被观察者观察到。
因此,KVC是KVO的基础。
Demo
比如我自定义的一个button
[self addObserver:self forKeyPath:@"highlighted"options:0 context:nil];
#pragma mark KVO
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
if([keyPath isEqualToString:@"highlighted"] ) {
[self setNeedsDisplay];
}
}
对于系统是根据keypath去取的到相应的值发生改变,理论上来说是和kvc机制的道理是一样的。
KVC机制通过key找到value的原理
当通过KVC调用对象时,比如:[self valueForKey:@”someKey”]时,
程序会自动试图通过下面几种不同的方式解析这个调用。
首先查找对象是否带有 someKey 这个方法,
如果没找到,会继续查找对象是否带有someKey这个实例变量(iVar),
如果还没有找到,程序会继续试图调用 -(id) valueForUndefinedKey:这个方法。
如果这个方法还是没有被实现的话,程序会抛出一个NSUndefinedKeyException异常错误。
补充:
KVC查找方法的时候,
不仅仅会查找someKey这个方法,
还会查找getsomeKey这个方法,
前面加一个get,或者_someKey以_getsomeKey这几种形式。
同时,查找实例变量的时候也会不仅仅查找someKey这个变量,
也会查找_someKey这个变量是否存在。
设计valueForUndefinedKey:方法的主要目的是
当你使用-(id)valueForKey方法从对象中请求值时,
对象能够在错误发生前,有最后的机会响应这个请求。
在 Objective-C 中如何实现 KVO
注册观察者(注意:观察者和被观察者不会被保留也不会被释放)
- (void)addObserver:(NSObject *)observer
forKeyPath:(NSString *)keyPath
options:(NSKeyValueObservingOptions)options
context:(void *)context;
接收变更通知
- (void)observeValueForKeyPath:(NSString *)keyPath
ofObject:(id)object
change:(NSDictionary *)change
context:(void *)context;
移除对象的观察者身份
- (void)removeObserver:(NSObject *)observer
forKeyPath:(NSString *)keyPath;
KVO中谁要监听谁注册,然后对响应进行处理,使得观察者与被观察者完全解耦。
KVO只检测类中的属性,并且属性名都是通过NSString来查找,
编译器不会检错和补全,全部取决于自己。
网友评论