前言

 

在上一阶段的开发过程中,我们大量使用了 KVO 机制,来确保页面信息的及时同步。也因此碰到了很多问题,促使我们去进一步学习 KVO 的相关机制,再到寻找更好的解决方案。鉴于 KVO 让人欲仙欲死的使用经历,在这里做一个简单分享。此分享的目的,更多的是在于点出 KVO 相关的技术点,供我们大家在学习和使用过程中做一个参考。

对于 KVO 的背后机制感兴趣的同学,可以直接看第三部分,KVC 和 isa-swizzling 。

对于 替代方案感兴趣的同学,请直接跳到末尾的第五部分,有列出了目前 github 上使用广泛的几个开源项目,它们让 KVO 变的更易用,总有一款适合你。如果各位有好的推荐,也请务必在评论里告诉我们,不胜感激。

对集合对象的观察,会在下次更新时做一个更具体的补充。

如果此文中有不当或错误的地方,也请各位批评指正,非常感谢~  没说的,报上工位号,请喝可乐~

一、什么是 KVO

键值观察是 Objective-C 语言的动态语言特性,在运行时通过 KVO,允许一个对象观察另一个对象的属性,当变化发生时,观察者会得到通知。

键值观察实际上是观察者模式在 Objective-C 中的一种运用,理解了观察者模式,也就理解了键值观察。

a)观察者和被观察对象完美分离

b)保持信息同步

二、通过一个示例了解 KVO 的基本用法

使用 KVO 的过程基本上分为三步:注册—通知—取消注册

下面来看一个示例。

某人养了一只宠物狗,这只宠物狗非常的聪明,会做加法题,而且很快。

我们在这里用 Person 来表示某人,Dog 来表示这只宠物狗,Person 有两个属性分别表示加数和被加数。

        @property (nonatomic, assgin) int numberOne;
        @property (nonatomic, assgin) int numberTwo;

1、Dog注册成为观察者,Person 为被观察对象,观察的属性为 numberOne、numberTwo

Dog:

        [person addObserver: self 
                 forKeyPath: @“numberOne”
                    options: NSKeyValueObservingOptionNew 
                    context: nil];     
 
        [person addObserver: self 
                 forKeyPath: @“numberTwo”
                    options: NSKeyValueObservingOptionNew 
                    context: nil];

2、Person 出题,Dog 收到通知后回答题目

Person 出题并发出通知:

        self.numberOne = 2;
        self.numberTwo = 3;

Dog 收到通知,回答题目:

        - (void) observeValueForKeyPath: (NSString *)keyPath 
                               ofObject: (id)object 
                                 change: (NSDictionary *)change 
                                context: (void *)context {
          // 根据 keyPath  判断是加数还是被加数发生变化,从 change 中获取新值,计算结果
        }

3、取消注册

Dog:

        [person removeObserver:self forKeyPath:@(numberOne)];
        [person removeObserver:self forKeyPath:@(numberTwo)];

三、KVO 原理详解

要理解 KVO 的原理,实际上也就是要搞清楚被观察对象在属性发生变化时,是如何做到通知观察者的。

这里面包含有两个点,一个是对属性的读取,一个是通知。

1、属性读取

说到对属性的读取,就不得不提 KVC,key-value coding,键值编码。实际上这也是 KVO 的基础。

KVC 提供了一种通过字符串标识符间接访问对象属性的机制。

1)支持这种机制的基本方法是:

        - (id) valueForKey:(NSString *)key;
        - (void) setValue:(id)value forKey:(NSString *)key;

例如访问 Person 对象的 numberOne 属性,可以通过以下方法实现:

        [person valueForKey:@“numberOne”];
        [person setValue:@(1) forKey:@“numberOne”];

对于实现了访问器方法的类来说,通过访问器方法(点语法)和通过 KVC 访问属性区别不大。但是对于没有实现访问器方法的类来说,点语法不可用,但是我们仍然可以通过 KVC 来访问属性。

下面来具体看下 KVC 访问属性时发生了什么。

KVC为了能设置和返回对象属性,会按照如下顺序进行尝试:

a)检查是否存在 - <key>,- is<Key> (只对布尔型有效),- get<Key> 的访问器方法,如果存在,则使用这些方法返回属性值。

检查是否存在 - set<Key> 的访问器方法,如果存在,则使用这些方法设置属性值。

b)如果上述方法不可用,则检查  - _<key>,- _is<Key> (只对布尔型有效),- _get<Key>,- _set<Key> 方法是否可用。

c)如果没有找到上述方法,会尝试直接访问实例变量,实例变量名可以是 <key> 或 _<key>

d)如果仍未找到,则调用 - valueForUndefinedKey: 和 - setValue:forUndefinedKey: 方法。这些方法的默认实现是抛出异常,我们可以根据需要进行重写。

由此我们也可以看出,当属性读取方法的定义符合命名规范的时候,KVC 能够定位到 键 key 对应的属性读取方法。

        // 属性访问器命名规范:     
        - (type) name;
        - (void) setName:(type)newName;
         
        // 特殊的:
        - (BOOL) isHidden;
        - (void) setHidden:(BOOL)newHidden;

2)除了基本方法之外,KVC 还提供了如下方法来支持通过键路径访问嵌套对象的属性

        - (id) valueForKeyPath:(NSString *)keyPath;
        - (void) setValue:(id)value forKeyPath:(NSString *)keyPath;

以及其它的一些方法,来支持对多关系的属性的读取。

* 对多关系的属性的读取,请参考 KVC 的相关文档

2、在属性读取方法里面,通知被观察者

这一步的实现,是基于 isa-swizzling (指针变化) 技术。

1)isa 是对象的一个特定指针,它指向对象的类, 该类中包含一张调度表,反映出选择器和最终实现之间的映射关系。当某个对象被第一次观察时,系统会在运行期动态创建一个派生类,isa 会指向这个新诞生的派生类。

例如我们之前的例子中,Person 对象的 isa 指针 在观察之前会指向 Person,在观察之后会指向 NSKVONotifying_Person

* 关于指针变换技术,大家可以参考 method swizzling http://www.cocoachina.com/applenews/devnews/2014/0225/7880.html

* 这个映射关系是可以更改的,涉及到 objective-c 的运行时技术,objc/runtime.h

* isa 指针的变化大家可以在代码中设置断点观察到

2)派生类会重写基类中任何被观察属性的 setter 方法, 真正的通知机制,正是在这个被重写的 setter 方法里面实现的。

例如:

        // 之前
      
        - (void) setNumberOne:(int)numberOne {
            _numberOne = numberOne;
        }
      
         
        //之后
        - (void) setNumberOne:(int)numberOne {
            [self willChangeValueForKey:@“numberOne”];
            _numberOne = numberOne;
            [self didChangeValueForKey:@“numberOne”];
        }

3、使用 KVO 通知观察者方法小结

1)使用 KVC 方法

如果有访问器方法,则运行时会在访问器方法中调用 will/didChangeValueForKey: 方法;

没用访问器方法,运行时会在 setValue:forKey 方法中调用 will/didChangeValueForKey: 方法。

2)使用访问器方法

运行时会重写访问器方法调用 will/didChangeValueForKey: 方法。因此,直接调用访问器方法改变属性值时,KVO也能监听到。

3)在赋值前后,手动调用 will/didChangeValueForKey: 方法。

四、实践

1、特点总结:

1)优点

  • 提供了一种简单方法让对象之间保持信息同步。例如模型对象和视图对象

  • 能够让我们观察某个对象的状态变化,即便该对象不是由我们创建的,也不能更改状态属性的实现方法

  • 观察对象可以了解该属性值新值以及旧值;如果观察的属性为对多的关系(例如数组),它也能够了解是哪个包含的对象发生了改变

  • 能够使用键路径 keypath 观察嵌套对象的属性变化

  • 彻底的抽象化,一个对象并不需要额外的代码来让自己变成可被观察对象

  • 多个 KVO 观察者可以观察同一对象的同一属性

2)缺点

  • 必须用字符串来指定要观察的属性,因此如果出错,在编译时是不会有检查和警告的

  • 重构对象属性之后,相关的 KVO 代码将不再起作用,由于不会有编译时自动检查,这部分代码甚至会引起崩溃

  • KVO 通知会触发一个特定的观察方法,观察必须要实现该方法,当观察者在观察多个属性时,在该方法中要写复杂的 if else 语句进行判断

  • 对象在销毁时要移除注册过的观察者

2、实际使用过程中,要特别注意的要点

1)在调用注册方法时传入适当的参数

        - addObserver:forKeyPath:options:context:

options:

功  能
NSKeyValueObservingOptionNew 作为变更信息的一部分发送新值
NSKeyValueObservingOptionOld 作为变更信息的一部分发送旧值
NSKeyValueObservingOptionInitial 在观察者注册时发送一个初始更新
NSKeyValueObservingOptionPrior 在变更前后分别发送变更,而不只在变更后发送一次

属性值的新值和旧值相同时,仍然能够触发 KVO,我们在注册时知道 new 和 old,能够让我们在通知方法中判断新旧值是否相同。

initial 可以确保在注册的同时,就触发一次 KVO 通知。

context:

由于我们无法指定通知方法,当在有通知发生时,如果子类和父类都实现了该方法,那么子类在处理通知时,无法通过 keyPath 和 object 来准确判断父类是否对该通知感兴趣,这个时候就需要子类父类在注册时根据需要传入不同的 context

* 关于 context 的最佳实践可以参考 http://stackoverflow.com/questions/12719864/best-practices-for-context-parameter-in-addobserver-kvo

2)手动触发 KVO

KVO 协议提供一个方法来关闭自动 KVO 通知:

    + (BOOL) automaticallyNotifiesObserversForKey:(NSString *)key

返回值为 NO 时,我们无论是通过 KVC 方法,还是访问器方法,都不会触发 KVO,我们需要自己手动调用will/didChangeValueForKey: 方法

3)使用单个 key 观察多个属性的变化

通过重写以下 方法,可以仅注册单个键的观察多个属性的变化:

    + (NSSet *) keyPathsForValuesAffectingValueForKey:(NSString *)key

4)KVO 支持对集合对象(NSArray、NSSet、NSOrderedSet)的观察,来及时获得集合内元素发生的变化,例如集合元素的增加、删除等等,变化的类型和具体内容会包含在通知方法的 change 字典中。

例如我们用 array 作为 tableView 的数据源,使用 kvo 方式观察 array 的变化,来自动触发 tableView 的刷新。

值得注意的是,我们没有办法使用 key path 来观察集合内部某元素的属性变化,要做到这一点,我们需要在往集合内添加和删除元素时,为每个元素单独注册和取消注册 KVO。

5)在通知方法中要注意线程问题

通知方法在哪个线程中被调用,是由被观察对象在哪个线程中触发 kvo 决定的。

6)不正确的取消注册会导致程序崩溃

a)不能重复取消相同的注册

b)如果是类似 @“a.b.c”  键路径,在取消注册时,a b 对象应当是存在的 。

针对原则 a,大家可以自行思考可以用什么方法来避免重复的注册和取消注册;

针对原则 b,需要注意的是我们在取消注册时,键路径中的对象是不是已经被释放了。

基于这两个原则,对于某些 UI 对象,除了考虑在 dealloc 中要取消注册外,还要根据实际情况来判断具体在什么位置注册和取消注册。以下是我在使用过程中遇到的,需要思考是否有必要做 注册和取消注册 的一些方法。

        // 复用的 cell
        - (void) prepareForReuse
         
        // 非复用的 view
        - (void) willMoveToWindow:(UIWindow *)newWindow
        - (void) didMoveToWindow
                
        - (void) didMoveToSuperview
        - (void) willMoveToSuperview:(UIView *)newSuperview
         
        // view controller     
        - (void) willMoveToParentViewController:(UIViewController 
        - (void) didMoveToParentViewController:(UIViewController *)parent*)parent
        - (void) viewDidLoad

五、更易用的 KVO

有一些开源项目,对 KVO 进行了二次封装,让 KVO 变的更易用,更安全,下面列举一些使用较广泛的供大家参考。在项目页面上,已经有了详细的特点说明和使用方法。

1)https://github.com/facebook/KVOController  推荐使用

2)https://github.com/th-in-gs/THObserversAndBinders

3)https://github.com/mikeash/MAKVONotificationCenter

六、参考文档:

Key-Value Observing Programming Guide

Key-Value Coding Programming Guide

http://blog.csdn.net/wzzvictory/article/details/9674431

http://blog.csdn.net/kesalin/article/details/8194240

http://www.cnblogs.com/lwzz/archive/2013/04/25/3029679.html

https://www.mikeash.com/pyblog/key-value-observing-done-right.html

深度理解Key-Value Observing 键值观察的更多相关文章

  1. K-V-O 键值观察机制

    在两个不同的控制器之间传值是iOS开发中常有的情况,应对这种情况呢,有多种的应对办法.kvc就是其中的一种,所以,我们就在此解释之.   key value observing  键值观察,给人一种高 ...

  2. xcode KVC:Key Value Coding 键值编码

    赋值 // 能修改私有成员变量 - (void)setValue:(id)value forKey:(NSString *)key; - (void)setValue:(id)value forKey ...

  3. [深入浅出Cocoa]详解键值观察(KVO)及其实现机理

    一,前言 Objective-C 中的键(key)-值(value)观察(KVO)并不是什么新鲜事物,它来源于设计模式中的观察者模式,其基本思想就是: 一个目标对象管理所有依赖于它的观察者对象,并在它 ...

  4. iOS - KVO 键值观察

    1.KVO KVO 是 Key-Value Observing 的简写,是键值观察的意思,属于 runtime 方法.Key Value Observing 顾名思义就是一种 observer 模式用 ...

  5. 《苹果开发之Cocoa编程》键-值编码和键-值观察

    一.KVC 键-值编码(Key - Value Coding, KVC)是通过变量名的读取和设置变量值的一种方法,将字符串的变量名作为key来引用.NSObject定义了两个方法(KVC方法)用于变量 ...

  6. OC键值观察KVO

    什么是KVO? 什么是KVO?KVO是Key-Value Observing的简称,翻译成中文就是键值观察.这是iOS支持的一种机制,用来做什么呢?我们在开发应用时经常需要进行通信,比如一个model ...

  7. 路径(keyPath)、键值编码(KVC)和键值观察(KVO)

    键路径 在一个给定的实体中,同一个属性的所有值具有相同的数据类型. 键-值编码技术用于进行这样的查找—它是一种间接访问对象属性的机制. - 键路径是一个由用点作分隔符的键组成的字符串,用于指定一个连接 ...

  8. [原创]obj-c编程17:键值观察(KVO)

    原文链接:[原创]obj-c编程17:键值观察(KVO) 系列专栏链接:objective-c 编程系列 说完了前面一篇KVC,不能不说说它的应用KVO(Key-Value Observing)喽.K ...

  9. obj-c编程17:键值观察(KVO)

    说完了前面一篇KVC,不能不说说它的应用KVO(Key-Value Observing)喽.KVO类似于ruby里的hook功能,就是当一个对象属性发生变化时,观察者可以跟踪变化,进而观察或是修正这个 ...

随机推荐

  1. redhat enterprixe 5.0 DNS 服务配置与管理

    一.了解DNS相关概念 DNS是一个分布式数据库,在本地负责控制整个分布式数据库的部分段,每一段中的数据通过客户机/服务器模式在整个网络上存取.通过采用复制技术和缓存技术使得整个数据库稳定可靠的同时, ...

  2. 在完成一个异步任务后取消剩余任务(C#)

    完整实例 using System;using System.Collections.Generic;using System.Linq;using System.Text;using System. ...

  3. L1 - 运行机制

    var name = 'kl'; function person(){ alert(name); var name = 'ko'; } person(); 这段代码输出 ‘undefined’,这种现 ...

  4. 一个QQ木马的逆向分析浅谈(附带源码)

    程序流程:首先注册自己程序的窗口以及类等一系列窗口操作,安装了一个定时器,间隔为100ms,功能搜索QQ的类名,如果找到就利用FindWindow("5B3838F5-0C81-46D9-A ...

  5. MSMQ消息队列

    MSMQ全称MicroSoft Message Queue,微软消息队列,是在多个不同的应用之间实现相互通信的一种异步传输模式,相互通信的应用可以分布于同一台机器上,也可以分布于相连的网络空间中的任一 ...

  6. js和C#中的编码和解码

    同一个字符串,用URL编码和HTML编码,结果是完全不同的. JS中的URL编码和解码.对 ASCII 字母和数字及以下特殊字符无效: - _ . ! ~ * ' ( ) ,/?:@&=+$# ...

  7. C语言中数组的几种输入

  8. matlab调用opencv函数的配置

    环境: VS2010 活动解决方案平台x64 WIN 8.1 Opencv 2.4.3 Matlab 2012a 1.  首先保证vs2010能正确调用opencv函数, 2.  Matlab中选择编 ...

  9. C#基础之程序集(一)

    一.什么是程序集? 程序集 其实就是bin目录的.exe 文件或者.dll文件. 二.原理 三.程序集分类 1.系统程序集 路径:C:\Windows\assembly 2.源代码生成的程序集 使用V ...

  10. 【django入门教程】Django的安装和入门

    很多初学django的朋友,都不知道如何安装django开发以及django的入门,今天小编就给大家讲讲django入门教程. 注明:python版本为3.3.1.Django版本为1.5.1,操作系 ...