3.4 KVO模型
KVO(key-value observer,“键-值”观察)模型是Cocoa绑定技术中常用的一种编程模型,它可以使一个对象在属性值发生变化时主动通知另一个对象并触发相应的方法。与NSNotification 不同,KVO没有所谓的中心对象来为所有观察者提供变化通知。当“被观察者”对象状态发生变化时,通知被直接发送至“观察者”对象,如图3-1所示。
在图3-1中,观察者是aButton,被观察者是modelObject。aButton是一个按钮控件,它是一个UI对象,有一个BOOL型的enabled属性,表示按钮是否可被点击。modelObject对象是一个模型对象,它没有可以呈现给用户的界面,同样,它有一个BOOL属性selected。通过KVO模型,modelObject的selected属性可以绑定到UIButton的enabled属性。即当modelObject的selected属性发生变化时(这是可以被编程的),KVO会主动通知aButton这种改变,因此按钮的外观随之可发生相应的呈现。比如由不可点击的灰色改变为可点击的着色状态。
KVO是一种很有用的绑定技术(Cocoa还提供另外一种绑定技术:Dynamic bingding)。而且它是由被观察的对象主动通知观察者的,并不需要经过一个统一的通知中心(如后面章节介绍的通知技术所述),它的执行效率和适用场景要更佳。
为了实现KVO,你需要进行如下操作:
注册观察者。所谓观察者即对象状态变化时需要通知的对象。
接收变更通知。接收变更通知主要是让观察者实现指定方法,在指定方法中,你可以接收到对象状态变更的消息,并在方法中进行处理。
取消所注册的观察者。观察者处理完状态变更消息之后,需要取消原先的注册状态。
3.4.1 注册KVO
对象要将自己注册为观察者,必须发送一个addObserver:forKeyPath:options:context:消息至被观察对象:
[account addObserver:inspector
forKeyPath:@"name"
options:(NSKeyValueObservingOptionNew |
NSKeyValueObservingOptionOld)
context:NULL];
以上例子将inspector对象注册为account对象的观察者,并表明观察者将对名为“name” 的属性变更感兴趣。
forKeyPath参数“name”注明了需要观察的属性的关键路径KeyPath。关键路径KeyPath实际是一个字符串,用于表示某个属性,你可以直接用属性名。但如果某属性是一个对象,则KeyPath可以用“.”语法的形式表示对象成员,如“account.name”。
options参数注明了对该属性的何种状态感兴趣。NSKeyValueObservingOptionNew表示属性在变更后的新值,NSKeyValueObservingOptionOld表示属性未改变之前的值。以上例子中的option参数设置表明,当name属性变更时,会将这两个值以NSDictionary的方式(即change参数)提交给观察者,观察者可以从NSDictionary中以键-值对的方式检索到这两个值。
context参数用于传递一个对象,该对象(或指针)会在属性变化时通过变更通知传递给观察者(通过context参数)。
移除观察者的注册,使用方法removeObserver forKeyPath:
[subject removeObserver:observer forKeyPath:@"name"];
3.4.2 接收变更通知
观察者要想收到对象的属性变更通知,需要实现方法observeValueForKeyPath:ofObject: change:context:,并在其中进行通知的处理。例如:
- (void)observeValueForKeyPath:(NSString )keyPath
ofObject:(id)object
change:(NSDictionary )change
context:(void )context
{
NSLog(@"%@",keyPath);
if ([keyPath isEqual:@"name"]) {
NSLog(@"name is changed:%@",
[change objectForKey:NSKeyValueChangeNewKey]);
}else
[super observeValueForKeyPath:keyPath
ofObject:object
change:change
context:context];
}
3.4.3 发送变更通知
NSObject支持两种属性变更通知,一种是自动变更通知,一种是手动变更通知。一般情况下,使用自动变更通知则更为简单,因此我们主要介绍自动变更通知。
1.自动变更通知
要使用自动变更通知,需要实现被观察者的 automaticallyNotifiesObserversForKey方法,在此方法中明确说明需要使用自动变更通知的属性。对于需要使用自动变更通知的属性,返回YES,如下代码所示:
+ (BOOL) automaticallyNotifiesObserversForKey:(NSString)key
{
// 对于属性name ,使用自动通知
if ([key isEqualToString:@"name"])
{
return YES;
}
// 确保调用了父类的automaticallyNotfiesObserversForKey方法
return [super automaticallyNotifiesObserversForKey:key];
}
然后,在name属性发生变化的时候通知观察者,比如调用以下语句之一:
subject.name=newName;
[subject setValue:newName forKey:@"name"];
[subject setValue:newName forKeyPath:@"name"];
如果属性是集合类型,则可以使用方法mutableSetValueForKey来支持以下集合方法导致的自动变更通知:
添加:insertObject:InKey:或者insertObject:AtIndex:
替换:replaceObject:InKey:或者replaceObject:AtIndex:
删除:removeObjectFromKey:或者removeObjectAtIndex:
2.手动变更通知
对于手动变更通知,除了需要在automaticallyNotifiesObserversForKey:方法中将要使用的手动变更通知返回NO外,还需要 在改变值之前调用willChangeValueForKey:并在更改它之后调用didChangeValueForKey:。
为了便于你理解KVO模型,我做了一个示例程序,放在光盘“source/第3章/TestKVO”文件夹,它使用了本节所介绍的知识点,可供参考和学习。