KVO的简单实用:
kvo是iOS中观察者模式的一种,他可以监听一个属性,在这个属性发生变化的时候,发出一个通知.
为什么使用KVO?
当我们根据一个对象的某一个属性发生变化时,改变另一个对象类的某个属性的时候,KVO就派上了用场.
KVO
KVO(Key-Value Observing) 是Cocoa提供的一种基于KVC的机制,允许一个对象去监听另外一个对象的某个属性.当该属性发生改变的时候,系统就会去通知监听的对象(就是谁在监听);
KVO的实现:
1.添加监听者.
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
这个方法的接收者,就是调用这个方法的对象,即观察者模式中的被观察者.有一点需要注意,被监听者的对象最好不要设置为当前上下文中容易发生改变的对象.因为KVO会吧添加KVO时的被监听者保存起来,当这个对象发生变化时,KVO监听的还是之前的那个对象,那么就不会去触发回调.
这里重点说下这几个参数:
observer: _就是要添加的监听者对象,当监听的属性发生改变时就会调用observeValueForKeyPath:ofObject:change:context:
,所以该监听者必须首先这个方法.要不然程序会抛出一个异常.
keyPath:_就是被监听的属性,这里和kvc一样不能是nil,否则会报错.通常我们在用的时候会写一个和属性一样的字符串,但注意不要拼写错误.(你可以写成NSStringFromSelector(@selector(propertyName))
这样.)
options:_这是一个配置选项,用来指明通知发出的时机和通知响应方法- observeValueForKeyPath:ofObject:change:context:
的change
字典中包含哪些值,这是一个枚举,一共有4个选项.可以选择多个,用|
符号连接
1> NSKeyValueObservingOptionNew:指明接受通知方法参数中的change字典中应该包含改变后的新值。
2>NSKeyValueObservingOptionOld: 指明接受通知方法参数中的change字典中应该包含改变前的旧值。
3>NSKeyValueObservingOptionInitial: 当指定了这个选项时,在addObserver:forKeyPath:options:context:
消息被发出去后,甚至不用等待这个消息返回,监听者对象会马上收到一个通知。这种通知只会发送一次,你可以利用这种“一次性“的通知来确定要监听属性的初始值。当同时制定这3个选项时,这种通知的change
字典中只会包含新值,而不会包含旧值。虽然这时候的新值实际上是改变前的'旧值',但是这个值对于监听者来说是新的。
4>NSKeyValueObservingOptionPrior:当指定了这个选项时,在被监听的属性被改变前,监听者对象就会收到一个通知(一般的通知发出时机都是在属性改变后,虽然change字典中包含了新值和旧值,但是通知还是在属性改变后才发出),这个通知会包含一个NSKeyValueChangeNotificationIsPriorKeykey,其对应的值为一个NSNumber类型的YES。当同时指定该值、new和old的话,change字典会包含旧值而不会包含新值。你可以在这个通知中调用- (void)willChangeValueForKey:(NSString )key;
** context*_最后一个参数是一个可选的参数,也就是可以不用填写,直接写一个nil.这个参数会被传到监听者的响应方法中去.可以用来传值,也可以用来区分不同点额通知.
需要注意的是,添加监听的方法
addObserver:forKeyPath:options:context:
并不会对监听和被监听的对象以及context做强引用,你必须自己保证他们在监听过程中不被释放。
响应方法:
- (void)observeValueForKeyPath:(nullable NSString *)keyPath ofObject:(nullable id)object change:(nullable NSDictionary<NSKeyValueChangeKey, id> *)change context:(nullable void *)context;
keyPath,object,context和监听方法中指定的一样,关于change参数,它是一个字典,有五个常量作为它的键:
NSString *const NSKeyValueChangeKindKey;
NSString *const NSKeyValueChangeNewKey;
NSString *const NSKeyValueChangeOldKey;
NSString *const NSKeyValueChangeIndexesKey;
NSString *const NSKeyValueChangeNotificationIsPriorKey;
移除监听
当一个监听者完成了它的监听任务之后,就需要注销(移除)监听者,调用以下2个方法来移除监听。通常会在-dealloc
方法或者-observeValueForKeyPath:ofObject:change:context:
方法中移除。
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(nullable void *)context
或者
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;