使用KVO键值监听
本文章从五个方面介绍KVO(Key-Value-Observer)键值观察者:
(1)功能介绍
(2)使用步骤
(3)应用场景
(4)原理理解
(5)相关的面试题
一 功能介绍
- KVO是OC语言对「观察者设计模式」的一种实现。
- 只要是NSObject的子类的实例对象,利用KVO机制可以监听该对象的指定属性的值,当属性值发生变化的时候,监听者就能获得通知,就能作出相应的处理。
- KVO触发机制:一个对象(观察者),监听另一个对象(被观察者)的某属性是否发生变化,若被监听的属性发生了更改,会触发观察者的一个方法(方法名是固定的,类似于代理方法)
- 使用KVO的好处之一是,不需要给被观察的实例对象添加任何额外的代码,就能对其实施监听。
- KVO的实现依赖于OC的运行时系统。
二 使用步骤
1、注册观察者
[<#要观察的实例对象#> addObserver:<#作为观察者的实例对象#> forKeyPath:<#要观察的实例对象的指定属性名#>
options:<#选项#> context:<#上下文,可以为KVO的回调方法传值#>];
关于options参数
typedef NS_OPTIONS(NSUInteger, NSKeyValueObservingOptions) {
//把更改之前的值提供给监听者的回调方法
NSKeyValueObservingOptionNew ,
//把更改之后的值提供给监听者的回调方法
NSKeyValueObservingOptionOld ,
//把初始化的值提供给监听者的回调方法,一旦注册立马就会调用一次。通常它会带有新值,而不会带有旧值
NSKeyValueObservingOptionInitial ,
//分两次调用监听者的回调方法。在值改变之前和值改变之后
NSKeyValueObservingOptionPrior
};
2、在回调方法中处理属性发生的变化
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change
(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context { if ([keyPath isEqualToString:@"name"]) {
id oldValue = [change objectForKey:@"old"]; id newValue = [change objectForKey:@"new"]; NSLog(@"%@~%@~%@", change, oldValue, newValue);
}
}
3、触发回调方法
使用setter方法或者KVC更改监听对象的指定属性,才会触发KVO。
4、移除观察者
注册观察者和移除观察者一定要成对出现。可以在dealloc中移除观察者,也可以在适当的时候提前手动移除观察者。
- (void)dealloc {
[self removeObserver:self forKeyPath:@"name"];
}
ps:一般使用KVO崩溃的原因:
(1)被观察的对象销毁了(比如被观察的对象被一个局部变量持有)
(2)观察者被释放掉了,但是没有移除监听(比如模态推出,push,pop等)
(3)注册的监听没有移除掉,又重新注册了一遍监听
三 应用场景
在MVC设计模式中,负责Model和View之间的同步。
比如,监听Model层,当Model层的数据发生更改时,同步View中的显示。
四 原理理解
比如我们要对Person类的实例对象person的name属性利用KVO机制实施监听属性值的变更通知。
第一步:运行时系统动态创建一个Person的子类:NSKVONotifying_Person
第二步: 将实例对象person的isa指针由原来的指向Person类,改为指向NSKVONotifying_Person类
第三步:改写NSKVONotifying_Person类中name属性的setter方法的实现:

(tips)-willChangevalueForKey:和-didChangeValueForKey:两个方法是KVO机制中向系统发送通知的实现。因此,上面提到的“触发回调方法”中,只说明了通过setter方法和KVC才能触动KVO机制,对属性直接赋值的方法是不能触发的,比如_name = @"XiaoMing";但是在了解KVO的原理之后,只要用上刚刚所知道的那两个方法,就行了。比如:

五 相关的面试题
(1)KVO与KVC的区别?
KVC是键值编码,提供了一种使用字符串直接访问和设置对象的属性的方式;
KVO是键值监听,提供了一种观察实例对象的指定属性的一种机制。
(2)KVO与Notifacation(通知)、delegate(代理)的区别?
Delegate的特点有:
1、1对1的传值
2、支持正向与反向传值(参数、返回值)
2、可以用require和optional来修饰
因此在使用delegate时,除了正常的通过参数传值,还可以灵活的运用返回值。
Notification的特点有:
1、1对多的传值
2、不关心返回值,单向的传值
3、NSNotificationCenter单例统一处理发通知
4、通过不同的唯一通知标识名NotificationName区分不同通知
5、被观察者主动发出通知
因此使用Notification一定要谨慎,由于1对多的缘故,避免滥用,不好查问题
其次就是注意要及时注销掉通知。比如:

KVO的特点有:
1、1对多
2、只能监听对象的属性变化,比较局限
3、只有通过KVC和setter方法才触发KVO
4、被观察者不用添加任何代码(比如由被观察者发出通知),观察者与被观察者完全解耦
5、注册观察者时,属性名都是通过字符串来查找,容易出错
6、KVO的要求是对象必须支持KVC机制(即必须是NSObject的子类)
7、也是属于单向传值
(3)KVO的优缺点?
优点:
1、能够提供一种简单的方法实现两个对象之间的同步,例如model和view之间的同步
2、能够对非我们创建的对象,即内部对象的状态作出响应,而且不需要改变内部对象的实现
3、能够提供观察的属性的最新值和先前值
4、使用的是keyPath来观察属性,因此也可以观察嵌套对象
5、完成了对观察对象的抽象,因为不需要额外的代码来允许观察值能够被观察
缺点:
1、我们观察的属性必须使用string来定义。因此在编译期不会出现警告以及检查
2、如果我们对属性进行了重构,那么之前KVO的代码需要重写
3、当释放观察者时需要移除观察者
4、KVO只能对属性作出反应,而不会对方法作出反应
使用KVO键值监听的更多相关文章
- KVO键值监听
KVO 键值监听:当我想知道一个对象的属性是否发生改变的时候 做出响应,就需要添加监听keypath 和 key的区别keypath会自动寻找子类的属性key只会寻找当前类的属性添加键值监听[pers ...
- ios中键值编码kvc和键值监听kvo的特性及详解
总结: kvc键值编码 1.就是在oc中可以对属性进行动态读写(以往都是自己赋值属性) 2. 如果方法属性的关键字和需要数据中的关键字相同的话 ...
- iOS监听模式系列之键值编码KVC、键值监听KVO的简单介绍和应用
键值编码KVC 我们知道在C#中可以通过反射读写一个对象的属性,有时候这种方式特别方便,因为你可以利用字符串的方式去动态控制一个对象.其实由于ObjC的语言特性,你根部不必进行任何操作就可以进行属性的 ...
- Unity3d 中键值监听方法
unity3d的api中没有负责监听键值的方法,不过unity的input类是通过c#类获取各类监听事件,所以我们可以通过c#类监听,方法如下: void OnGUI() { Event e = Ev ...
- KVO-键值监听
键值监听,就是可以监听对象某个属性值的变化: 首先,在工程中,新建一个Person的类 @interface Person : NSObject @property (nonatomic, copy) ...
- KVO 键值观察者
KVO(键值观察者) //监听的创建 -(id)initChildren:(Person *)person { self = [super init]; if (self != nil) { //拥有 ...
- Android应用中返回键的监听及处理
MainActivity: package com.testnbackpressed; import android.os.Bundle; import android.view.KeyEvent ...
- Android应用中Back键的监听及处理
MainActivity如下: package cn.testnbackpressed; import android.os.Bundle; import android.view.KeyEvent; ...
- Key-Value Observing (键值监測)
Key-Value Observing (键值监測) 简单介绍 KVO是一套当目标对象的属性值改变时观察者对象能够接受到通知的机制.必须先理解KVC才干更好的理解KVO,前者是后者的实现基础. 这种通 ...
随机推荐
- 507,介绍一下标准的css盒子模型?低版本ie的盒子模型有什么不同的?
有两种,IE盒子模型,另外是W3C盒子模型: 盒模型都包括:内容(content),填充(padding),边界(margin),边框(border): 区别:IE的content部分吧border和 ...
- 存储引擎:engine
1.表类型: 默认的服务器表类型,通过my.ini文件可以手动修改配置:default-storage- engine=INNODB 在创建表,或者编辑表时,可以指定表的存储引擎: 语法:engine ...
- 每天进步一点点------SOPC的Avalon-MM IP核(三) LCD1602 IP定制
注:Avalon信号类型命名参考图 /********************************************************************************* ...
- SpringBoot集成SwaggerUI
1.在module下的pom.xml中引用相关插件 引用swagger插件并用参数化版本信息,如下 <?xml version="1.0" encoding="UT ...
- spark-shell中往mysql数据库写数据报错
今天在看spark方面的知识的时候,在spark-shell中往mysql写数据时报错,错误信息如下: ERROR Executor: Exception in task 0.0 in stage 4 ...
- pyqt:布局删除小部件
参照:https://stackoverflow.com/questions/5899826/pyqt-how-to-remove-a-widget import sip layout.removeW ...
- 01hive基础操作
一. Hive基础概念 我自己本人一开始学习Hive的时候,最大的疑问就是hive和hbase到底有什么区别?(因为自己本身学校课程设置问题有了解到一丢丢hbase的知识) 所以先甩一篇博客提供给跟我 ...
- npm报错This is probably not a problem with npm. There is likely additional logging
使用webstorm开发时,遇到npm 报错,于是尝试了如下所有的方法,不完全统计. https://blog.csdn.net/liu305088020/article/details/791823 ...
- warmup
先简单了解下源码中的2个函数: <?php echo mb_strpos("朋友比生命还重要?或许是吧" . '?',"?"); echo "\ ...
- Jmeter BeanShell笔记
前言:beanshell是支持java语法的,因此当一些复杂的数据结构jmeter处理不了的时候,我们可以借助于java来实现 1,使用beanshell处理JDBC请求返回的值 数据库结构 当JDB ...