简介
KVO全称KeyValueObserving
,俗称键值监听,是苹果提供的一套事件通知机制。允许对象监听另一个对象特定属性的改变,并在改变时接收到事件。由于KVO的实现机制,所以对属性才会发生作用,一般继承自NSObject
的对象都默认支持KVO。
基本用法
@interface YZPerson : NSObject
@property (nonatomic,assign) int age;
@property (nonatomic,strong) NSString *name;
@end
- (void)viewDidLoad {
[super viewDidLoad];
self.person = [[YZPerson alloc] init];
// 注册观察者
NSKeyValueObservingOptions options = NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld;
[self.person addObserver:self forKeyPath:@"name" options:options context:@"1111"];
}
// 回调方法。当监听对象的属性值发生改变时,就会调用
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
NSLog(@"监听到%@的%@属性值改变了 - %@ - %@", object, keyPath, change, context);
}
-(void)dealloc{
// 移除监听
[self.person removeObserver:self forKeyPath:@"name"];
}
注意:重复移除监听会崩溃,因为已经移除了,找不到了。所以,要保证add和remove是成对出现的。
KVO可以监听单个属性的变化,也可以监听集合对象的变化。
通过KVC的mutableArrayValueForKey:
等方法获得代理对象,当代理对象的内部对象发生改变时,会回调KVO监听的方法。集合对象包含NSArray
和NSSet
。
底层原理
KVO是通过isa-swizzling技术实现的。
1.在运行时Runtime根据原类创建一个中间类NSKVONotifying_xxx
,这个中间类是原类的子类,并动态修改当前 对象 的isa
指向中间类。
2.当修改 对象的属性时,会调用 _NSSetXXXValueAndNotify
函数 。
该函数里面会先调用willChangeValueForKey:
,然后调用父类原来的 setter
方法修改值,最后是didChangeValueForKey:
,它的内部会触发监听器(Oberser)
的监听方法observeValueForKeyPath:ofObject:change:context:
,并且将class方法重写,返回原类的Class。
关于中间类
对同一个类的不同对象进行KVO,因为动态指向了各自的中间类,它们的类对象不同。所以可以准确监听不同对象的特定属性。
由[instance class]
得到的类名是一样的(即instance
所属的类),而不是中间类的名字。关于setter
当改变属性的时候,是通过调用它的setter进行的。添加KVO监听之后,setter 指向了_NSSetObjectValueAndNotify
函数。
如果没有通过 setter 来改变属性值,或者没有属性没有对应的setter方法,则不会触发KVO。手动触发KVO
手动调用对象的willChangeValueForKey、didChangeValueForKey
两个方法,也会触发监听回调方法,但是属性值不变。因为没有给属性赋值。
[self.person willChangeValueForKey:@"name"];
[self.person didChangeValueForKey:@"name"];
对比 KVC、NSNotificationCenter
KVC(键值编码),即Key-Value Coding
,一个非正式的Protocol
,使用字符串(键)访问一个对象实例变量的机制。而不是通过调用 setter、getter
方法等显式的存取方式去访问。
KVC和KVO都属于键值编程,而且底层实现机制都是isa-swizzing。
KVO和NSNotificationCenter都是iOS中观察者模式的一种实现。
两者都是一对多,但是对象之间直接的交互,notification 明显得多,需要notificationCenter 来做为中间交互。
notification 的优点是,监听不局限于对属性,还可以对多种多样的状态变化进行监听,监听范围广。
KVO对被监听对象无侵入性,不需要修改其内部代码即可实现监听。
其他
- 用KVC修改UIPageControl的选中图片和默认图片
(系统默认是不允许修改的)
[self.pageControl setValue:currentImage forKey:@"_currentPageImage"];
[self.pageControl setValue:pageImage forKey:@"_pageImage"];
- 如何对 NSMutableArray 进行 KVO
一般情况下只有通过调用 set 方法对值进行改变才会触发 KVO。但是在调用NSMutableArray
的addObject
或removeObject
系列方法时,并不会触发它的 set 方法。
为了实现NSMutableArray
的 KVO,官方为我们提供了如下方法:
@property (nonatomic, strong) NSMutableArray *arr;
//添加元素操作
[[self mutableArrayValueForKey:@"arr"] addObject:item];
//移除元素操作
[[self mutableArrayValueForKey:@"arr"] removeObjectAtIndex:0];