KVO,即Key-Value Observing,是 Objective-C 对观察者模式(Observer Pattern)的一种实现。也是 Cocoa Binding 的基础。当被观察对象的某个属性发生更改时,观察者对象会获得通知。
KVO的通知方式一般分为两种,一种是** 自动通知 ** ,另一种是** 手动通知 ,系统默认的是 自动通知 **。
KVO的实现步骤
首先我们要有一个类,在这篇文章的例子中,我们来对这个类的属性进行观察。
类的实现如下:
.h 文件
#import
@interface Person : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, copy) NSString *gender;
@end
.m 文件
#import "Person.h"
@implementation Person
@end
KVO的实现一般有以下三步
1. 添加观察,并指定被观察的属性
我们在程序启动时为Person类的实例对象添加观察者,即当前控制器。为了一会解释手动实现,我们把name和gender属性都设置为被观察的属性。且观察的是这两个属性的新值。
Person *person = [[Person alloc] init];
[person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew context:nil];
[person addObserver:self forKeyPath:@"gender" options:NSKeyValueObservingOptionNew context:nil];
self.person = person;
2. 实现观察方法
KVO观察模式在被观察属性发生变化时,会自动调用下面的方法,我们要在方法里实现一些逻辑。
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
if ([keyPath isEqualToString:@"name"]) {
NSLog(@"new name = %@", [change objectForKey:NSKeyValueChangeNewKey]);
} else if ([keyPath isEqualToString:@"gender"]) {
NSLog(@"new gender = %@", [change objectForKey:NSKeyValueChangeNewKey]);
}
}
3. 移除观察者
KVO会持有观察者,所以在必要的时候,我们要移除观察者。如果不移除,可能会造成访问僵尸对象的情况,比如刘爽同学当时为一个单利变量添加了一个控制器观察者,当这个控制器被pop掉后,单利中的值发生了改变,结果在再次发出通知时,这个控制器变量不存在,造成了访问僵尸对象,引起了奔溃。所以我们要移除观察者。
移除观察者一般是在观察者的dealloc方法里完成的。
- (void)dealloc
{
[self.person removeObserver:self forKeyPath:@"name"];
[self.person removeObserver:self forKeyPath:@"gender"];
}
自动通知的实现
现在,我们就要进行被观察属性的修改了,看看自动通知是怎么工作的。
person.name = @"Alan Walker";
person.gender = @"male";
person.name = @"DJ Earworm";
person.gender = @"male";
控制台上打印
可见,在KVO的自动通知中,只要是给被观察对象赋值,无论该值是新值还是旧值,都会发出通知。
手动通知的实现
要实现手动通知,我们需要对Person模型进行一些修改。
在Person中添加** + (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key; ** 方法
+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key
{
if ([key isEqualToString:@"name"]) {
return NO;
} else {
return [super automaticallyNotifiesObserversForKey:key];
}
}
这样,我们就关闭了name属性的自动通知。
1. 如果想要实现被观察属性在被赋的值与原来相同时,不进行通知,可以重写name的setter方法
- (void)setName:(NSString *)name
{
if (self.name != name) {
[self willChangeValueForKey:@"name"];
_name = name;
[self didChangeValueForKey:@"name"];
}
}
运行如下代码
person.name = @"Alan Walker";
person.gender = @"male";
person.name = @"DJ Earworm";
person.gender = @"male";
person.name = @"DJ Earworm";
person.gender = @"male";
打印结果如下
2. 我们还可以按照自己的意愿让通知启动。
撤销重写setter方法,执行如下代码
person.name = @"Alan Walker";
person.gender = @"male";
[person willChangeValueForKey:@"name"];
person.name = @"DJ Earworm";
[person didChangeValueForKey:@"name"];
person.gender = @"male";
打印结果如下
由此可见,手动实现通知以下两个方法比较重要。
- (void)willChangeValueForKey:(NSString *)key;
- (void)didChangeValueForKey:(NSString *)key;