一. KVO是键值观察,是Objective-C对观察者模式的实现,每次当被观察者对象的某个属性值发生改变时,注册的观察者便能获得通知
二.原理:当观察某对象A时,KVO机制动态创建一个对象A当前类的子类:NSKVONotifying_A,并为这个新的子类重写了被观察属性keyPath的setter方法。 setter方法随后负责通知对象属性的改变状况.
在这个过程中,被观察者对象的isa指针从指向原来的A类,被KVO机制修改为指向系统新创建的子类NSKVNitifying_A类,来实现当前类属性值改变的监听
isa指针的作用:每个对象都有isa指针,指向该对象的类,它告诉Runtime系统这个对象的类是什么。所以对象注册为观察者时,isa指针指向新子类,那么这个被观察的对象就变成新子类的对象,因而在该对象上对setter的调用就会调用已重写的setter,从而激活键盘值通知机制
KVO的键值观察通知依赖于NSObject的两个方法:willChangeValueForKey: 和 didChangeValueForKey
三.方法
1.注册观察者,实施监听
[person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];
其中person为被观察者对象,他的name属性即为被观察者的属性
2.在回调方法中处理属性发生的变化
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(voidvoid *)context
{
NSString *name = [object valueForKey:@"name"];
NSLog(@"new name is: %@", name);
}
只要person对象中的name发生变化,系统会自动调用该方法
3.移除观察者
-(void)dealloc
{
[person removeObserver:self forKeyPath:@"name"];
}
四.扩展
1.kvo与kvc不同
KVC (键值编码)即key-value coding ,一个非正式的协议,使用字符串(键)访问一个对象实例变量的机制。而不是通过调用setter getter 方法存取方式去访问。 运用了isa-swizzling技术,isa-swizzling就是类型混合指针机制。
SEL数据类型:它是编译器运行Objective-C里的方法的环境参数。
IMP数据类型:它其实就是一个编译器内部实现时候的函数指针。当Objective-C编译器去处理实现一个方法的时候,就会指向一个IMP对象,这个对象是C语言表达的类型
KVC内部实现:(1)首先根据方法名找到运行方法的时候所需要的环境参数。 (2)它会自己isa指针介乎环境参数,找到具体的方法实现的接口。(3)再直接查找得来的具体的方法实现
KVO(键值监听),即Key-Value Observing,它提供一种机制,当指定的对象的属性被修改后,对象就会接受到通知,前提是执行了setter方法、或者使用了KVC赋值。kvo是基于kvc实现的
2.和notification(通知)的区别?
notification比KVO多了发送通知的一步。
两者都是一对多,但是对象之间直接的交互,notification明显得多,需要notificationCenter来做为中间交互。而KVO如我们介绍的,设置观察者->处理属性变化,至于中间通知这一环,则隐秘多了,只留一句“交由系统通知”,具体的可参照以上实现过程的剖析。
3.与delegate的不同?
和delegate一样,KVO和NSNotification的作用都是类与类之间的通信。但是与delegate不同的是:
这两个都是负责发送接收通知,剩下的事情由系统处理,所以不用返回值;而delegate 则需要通信的对象通过变量(代理)联系;
delegate一般是一对一,而这两个可以一对多。
链接:http://blog.csdn.net/chy305chy/article/details/51793373 http://www.jianshu.com/p/e59bb8f59302 http://www.jianshu.com/p/37a92141077e