参考链接:
http://tech.glowing.com/cn/implement-kvo/ 手动实现KVO
1.addObserver:forKeyPath:options:context:各个参数的作用分别是什么,observer中需要实现哪个方法才能获得KVO回调?
// 添加键值观察
/*1 观察者,负责处理监听事件的对象
2 观察的属性
3 观察的选项
4 上下文
*/
[self.personaddObserver:selfforKeyPath:@"name"options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOldcontext:@"Person Name"];
observer中需要实现一下方法:
// 所有的 kvo 监听到事件,都会调用此方法
/*1. 观察的属性
2. 观察的对象
3. change 属性变化字典(新/旧)
4. 上下文,与监听的时候传递的一致
*/
- (void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)object change:(NSDictionary*)change context:(void*)context;
2.如何手动触发一个value的KVO
所谓的“手动触发”是区别于“自动触发”:
自动触发是指类似这种场景:在注册 KVO 之前设置一个初始值,注册之后,设置一个不一样的值,就可以触发了。
想知道如何手动触发,必须知道自动触发 KVO 的原理:
键值观察通知依赖于 NSObject 的两个方法:willChangeValueForKey:和didChangevlueForKey:。在一个被观察属性发生改变之前,willChangeValueForKey:一定会被调用,这就 会记录旧的值。而当改变发生后,observeValueForKey:ofObject:change:context:会被调用,继而didChangeValueForKey:也会被调用。如果可以手动实现这些调用,就可以实现“手动触发”了。
那么“手动触发”的使用场景是什么?一般我们只在希望能控制“回调的调用时机”时才会这么做。
具体做法如下:
如果这个value是 表示时间的self.now,那么代码如下:最后两行代码缺一不可。
// 手动触发 value 的KVO,最后两行代码缺一不可。
//@property (nonatomic, strong) NSDate *now;
- (void)viewDidLoad {
[superviewDidLoad];
_now = [NSDatedate]; [selfaddObserver:selfforKeyPath:@"now"options:NSKeyValueObservingOptionNewcontext:nil];NSLog(@"1"); [selfwillChangeValueForKey:@"now"];// “手动触发self.now的KVO”,必写。NSLog(@"2"); [selfdidChangeValueForKey:@"now"];// “手动触发self.now的KVO”,必写。NSLog(@"4");
}
但是平时我们一般不会这么干,我们都是等系统去“自动触发”。“自动触发”的实现原理:
比如调用setNow:时,系统还会以某种方式在中间插入wilChangeValueForKey:、didChangeValueForKey:和observeValueForKeyPath:ofObject:change:context:的调用。
大家可能以为这是因为setNow:是合成方法,有时候我们也能看到有人这么写代码:
- (void)setNow:(NSDate*)aDate {
[selfwillChangeValueForKey:@"now"];// 没有必要_now = aDate;
[selfdidChangeValueForKey:@"now"];// 没有必要
}
这完全没有必要,不要这么做,这样的话,KVO代码会被调用两次。KVO在调用存取方法之前总是调用willChangeValueForKey:,之后总是调用didChangeValueForkey:。