KVO-监听属性值

简介

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是成对出现的。markdown

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

  • 关于中间类

对同一个类的不一样对象进行KVO,由于动态指向了各自的中间类,它们的类对象不一样。因此能够准确监听不一样对象的特定属性。 由[instance class]获得的类名是同样的(即instance所属的类),而不是中间类的名字。spa

  • 关于setter

当改变属性的时候,是经过调用它的setter进行的。添加KVO监听以后,setter 指向了 _NSSetObjectValueAndNotify函数。 若是没有经过 setter 来改变属性值,或者没有属性没有对应的setter方法,则不会触发KVO。代理

  • 手动触发KVO

手动调用对象的willChangeValueForKey、didChangeValueForKey两个方法,也会触发监听回调方法,可是属性值不变。由于没有给属性赋值。code

[self.person willChangeValueForKey:@"name"];
[self.person didChangeValueForKey:@"name"];
复制代码

对比 KVC、NSNotificationCenter

KVC(键值编码),即 Key-Value Coding,一个非正式的 Protocol,使用字符串(键)访问一个对象实例变量的机制。而不是经过调用 setter、getter 方法等显式的存取方式去访问。orm

KVC和KVO都属于键值编程,并且底层实现机制都是isa-swizzing。server

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];
复制代码
相关文章
相关标签/搜索