KVO和手动调用KVO

参考链接:

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:。

你可能感兴趣的:(KVO和手动调用KVO)