KVO-监听属性值

2,152 阅读3分钟

简介

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监听的方法。集合对象包含NSArrayNSSet

底层原理

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 addObjectremoveObject 系列方法时,并不会触发它的 set 方法。 为了实现NSMutableArray的 KVO,官方为我们提供了如下方法:
@property (nonatomic, strong) NSMutableArray *arr;

//添加元素操作
[[self mutableArrayValueForKey:@"arr"] addObject:item];
//移除元素操作
[[self mutableArrayValueForKey:@"arr"] removeObjectAtIndex:0];