[iOS]KVO的相关知识

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";

控制台上打印

1.png

可见,在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.png
2. 我们还可以按照自己的意愿让通知启动。

撤销重写setter方法,执行如下代码

    person.name = @"Alan Walker";
    person.gender = @"male";
    
    [person willChangeValueForKey:@"name"];
    person.name = @"DJ Earworm";
    [person didChangeValueForKey:@"name"];
    person.gender = @"male";

打印结果如下

3.png
由此可见,手动实现通知以下两个方法比较重要。
- (void)willChangeValueForKey:(NSString *)key;
- (void)didChangeValueForKey:(NSString *)key;

你可能感兴趣的:([iOS]KVO的相关知识)