iOS开发-消息传递方式-KVO

上次我们说到了target-action

这次我们来说说另一种消息传递方式,KVO

KVO(Key-Value Observing),翻译过来就是键值监听,是观察者模式的一种实现方式,也就是监听某一个对象,当他发生改变时,通知另外一个对象,做出相应的动作。在MVC设计模式中,通常用于Model和Contorller的通讯(另外一种方式是Notification)。

打个比方就是,我看到冰箱里的牛奶没有了,我就要打电话给我妈妈让他买一点。

如何使用KVO

使用KVO的步骤比较简单,但是坑非常多,总体来说使用KVO一共需要4个步骤。

  • 1.注册观察者。
  • 2.实现回调方法。
  • 3.触发回调方法。
  • 4.移除观察者。

注册观察者

注册观察者通常通过函数

1
2
3
4
- (void)addObserver:(NSObject *)observer
forKeyPath:(NSString *)keyPath
options:(NSKeyValueObservingOptions)options
context:(nullable void *)context;

来实现,接下来我们看看每个参数的作用。

observer

指观察者对象,用之前买牛奶的🌰来说的话,这个就是我。

keyPath

指被观察者的属性,这个值不能为nil,这个值非常重要,后边的很多操作都和他有关,而且这是一个字符串,写错了的话,编译器也不会报错,很大的一个坑。

还有,这个值一定要是被观察的属性的名称,比如,我观察的是milk,这个属性的变化,那我的keyPath就一定要是milk。

options

指发出了通知之后带的信息,会包含在change字典中,由NSKeyValueObservingOptions中的4个值构成。

1
2
3
4
5
6
7
8
9
10
11
12
typedef NS_OPTIONS(NSUInteger, NSKeyValueObservingOptions) {
//表示change中包含观察对象的新值
NSKeyValueObservingOptionNew = 0x01,
//表示change中包含观察对象的旧值
NSKeyValueObservingOptionOld = 0x02,
//如果设定了该值,在注册观察者的方法返回之前就会发送通知给观察者。在注册观察者时,如果同时指定了 NSKeyValueObservingOptionNew,那么在发出的通知中, change 字典中会包含 NSKeyValueChangeNewKey 及被观察对象的当前值,但是却不会包含 NSKeyValueChangeOldKey,且 NSKeyValueChangeKindKey 对应的值是 NSKeyValueChangeSetting。
NSKeyValueObservingOptionInitial API_AVAILABLE(macos(10.5), ios(2.0), watchos(2.0), tvos(9.0)) = 0x04,
//使得被观察对象的值在改变之前和改变之后都会发送通知,而不仅仅是在改变之后发送一个通知。
NSKeyValueObservingOptionPrior API_AVAILABLE(macos(10.5), ios(2.0), watchos(2.0), tvos(9.0)) = 0x08
};

context

这个值可以在回调方法中传递给方法内容。

实现回调方法

注册的观察者之后,我们就可以先实现回调方法了。

1
2
3
4
5
6
7
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
{
if([keyPath isEqualToString:KVOPath])
{
}
}

因为在一个控制器中,所有的KVO回调都会走到这个方法中来,所以,在这个方法中一般都会加入一个判断来区分是哪个值发生了改变。

触发回调方法

一般情况下,当我们设定好监听之后,改变被监听对象的值时,就可以出发回调方法了。这里要注意一点,如果直接使用 _变量名来直接赋值是不会走setter方法的,所以这个时候不会出发KVO的回调方法,我们需要使用self.变量名来对变量值进行修改才会调用KVO的方法。

移除观察者

一般在一个页面即将销毁时,我们需要调用

1
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;

来移除观察者,一般在dealloc方法中调用他。

这个地方会出很多问题,比如,如果没有注册这个方法,这时候移除他就会出现异常。所以这里我们最好使用@try-catch来防止异常发生。
又或者在页面销毁的时候没有移除观察者,也会出现异常。

Demo

简单介绍了一下KVO的使用方法,接下来我们通过一个例子来使用一下KVO。

如果嫌看代码麻烦,也可以直接下Demo出来看。

建立被观察模型

首先我们创建一个model用来被观察。

1
2
3
4
5
6
7
8
#import <Foundation/Foundation.h>
@interface MPModel : NSObject
@property (nonatomic ,assign) int number;
@property (nonatomic ,copy) NSString *name;
- (instancetype)init;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
#import "MPModel.h"
@implementation MPModel
@synthesize number;
- (instancetype)init
{
self = [super init];
if(self)
{
self.number = 100;
self.name = @"okok";
}
return self;
}
@end

注册KVO

1
2
3
4
5
6
NSString *const KVOPath = @"number";
- (void)addKVO
{
[self.model addObserver:self forKeyPath:KVOPath options:NSKeyValueObservingOptionOld|NSKeyValueObservingOptionNew context:nil];
}

实现回调方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context
{
if([keyPath isEqualToString:KVOPath])
{
id oldName = [change objectForKey:NSKeyValueChangeOldKey];
NSLog(@"oldName----------%@",oldName);
id newName = [change objectForKey:NSKeyValueChangeNewKey];
NSLog(@"newName-----------%@",newName);
self.displayLabel.text = [NSString stringWithFormat:@"%ld",(long)self.model.number];
}
}

移除KVO

1
2
3
4
5
6
7
8
9
10
11
12
- (void)removeKVO
{
@try
{
[self.model removeObserver:self forKeyPath:KVOPath];
}
@catch(NSException *exception)
{
}
}

这样设定了之后,在改变number值的时候,就会触发回调方法,将新的number值显示在Label上。

KVO和Runtime

最后简单说一下KVO的实现,KVO是依赖于Runtime来实现的,当我们设定了一个观察对象的时候,一个新的类会被动态的创建出来,这个类继承了被观察类的属性,并重写了被观察属性的setter方法。重写时添加了相应的通知,最后把isa指针(isa指针告诉Runtime系统的这个对象的类是什么)指向这个新建的类。

参考文章

iOS初探KVO

深入理解KVO

KVC 和 KVO详解

本篇文章仅限个人学习使用,如果有什么不对的地方还请大佬们多多批评指正。