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

上次我们说到了target-action

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

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

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

如何使用KVO

使用KVO的步骤比较简单,但是坑非常多,总体来说使用KVO一共需要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个值构成。


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

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

实现回调方法

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


- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context

{

 if([keyPath isEqualToString:KVOPath])

 {

 }

}

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

触发回调方法

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

移除观察者

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


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

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

这个地方会出很多问题,比如,如果没有注册这个方法,这时候移除他就会出现异常。所以这里我们最好使用@try-catch来防止异常发生。

又或者在页面销毁的时候没有移除观察者,也会出现异常。

Demo

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

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

建立被观察模型

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


#import <Foundation/Foundation.h>

@interface MPModel : NSObject

@property (nonatomic ,assign) int  number;

@property (nonatomic ,copy) NSString *name;

- (instancetype)init;


#import "MPModel.h"

@implementation MPModel

@synthesize number;

- (instancetype)init

{

 self = [super init];

 if(self)

 {

 self.number = 100;

 self.name = @"okok";

 }

 return self;

}

@end

注册 KVO


NSString *const KVOPath = @"number";

- (void)addKVO

{

 [self.model addObserver:self forKeyPath:KVOPath options:NSKeyValueObservingOptionOld|NSKeyValueObservingOptionNew context:nil];

}

实现回调方法


- (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


- (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详解

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,776评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,527评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,361评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,430评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,511评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,544评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,561评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,315评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,763评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,070评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,235评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,911评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,554评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,173评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,424评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,106评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,103评论 2 352

推荐阅读更多精彩内容