1、KVO是基于Runtime机制实现的;

2、当某个类的对象的某个属性第一次被观察时,系统会在运行期间动态地创建该类的一个派生类,在这个派生类中重写基类的任何被观察属性的setter方法,派生类在被重写的setter方法内实现真正的通知机制;

3、如果原类为Person,那么生成的派生类名为NSKVONotifying_Person;

4、我们知道,每一个对象都有一个isa指针(即第一个成员变量)指向当前类,所以系统就是在当一个类的对象第一次被观察的时候,偷偷的将isa指针指向动态生成的派生类,从而在 “被监听属性” 赋值时执行的是派生类的setter方法;

5、此外,苹果还偷偷重写了此派生类的class方法,这时候 po personObj,打印的依旧是Person,但使用po object_getClass(personObj),就可以打印出对象的类型NSKVONotifying_Person;这是苹果为了隐藏生成的派生类,让我们误以为还是使用的当前类;

6、键值观察通知依赖于NSObject 的两个方法: willChangeValueForKey: 和 didChangevlueForKey:;在一个被观察属性发生改变之前, willChangeValueForKey: 一定会被调用,这就会记录旧的值;而当改变发生后,didChangeValueForKey: 会被调用,继而 observeValueForKey:ofObject:change:context: 也会被调用。

7、注意:willChangeValueForKey:和didChangevlueForKey:缺一不可;

直接拿代码来说明吧,说了一大堆,估计大家都看的不知所以然。

1、场景:我们声明了一个Person类,里面有两个属性,name和age,我们要使用kvo来监听Person对象的age发生的变化;

2、创建Person类,声明属性:

 #import <Foundation/Foundation.h>

 /**
Person模型类
*/
@interface Person : NSObject /**
姓名
*/
@property(nonatomic,copy) NSString *name; /**
年龄
*/
@property(nonatomic,assign) NSInteger age; @end

3、创建Person对象,添加KVO监听对象age属性

 #import "ViewController.h"
#import "Person.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad]; Person *per = [[Person alloc] init];
per.name = @"xiaoming";
per.age = ; // 观察per对象的age属性的变化情况
[per addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:NULL]; // 修改age属性的值,将会被观察到
per.age = ; // 这里po per的结果:<Person: 0x608000026960>
// 但是:po object_getClass(per)的结果:NSKVONotifying_Person
// 所以:苹果偷偷重写了派生类的class方法,但用运行时可以查看当前对象所属类型 // 移除KVO
[per removeObserver:self forKeyPath:@"age"];
} /**
KVO监听:当被监听对象的被监听属性变化后调用 @param keyPath 被监听的属性
@param object 被监听的对象
@param change 变化情况
@param context 携带的参数,此示例传入NULL即可
*/
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
NSLog(@"%@对象的%@属性发生变化了%@",object,keyPath,change); // 打印结果:
// <Person: 0x608000026960>对象的age属性发生变化了{
// kind = 1;
// new = 20;
// }
} @end

其实,实现KVO的是这2个方法:willChangeValueForKey:和didChangevlueForKey:

我们只要调用这两个方法,属性值不变化,也会被观察到。

1、给Person类添加一个方法kvoTest:

 #import <Foundation/Foundation.h>

 /**
Person模型类
*/
@interface Person : NSObject /**
姓名
*/
@property(nonatomic,copy) NSString *name; /**
年龄
*/
@property(nonatomic,assign) NSInteger age; /**
监听age属性发生变化
*/
- (void)kvoTest; @end
 #import "Person.h"

 @implementation Person

 - (void)kvoTest{

     [self willChangeValueForKey:@"age"];

     [self didChangeValueForKey:@"age"];
} @end

2、调用kvoTest方法监听对象属性age的变化:

 #import "ViewController.h"
#import "Person.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad]; Person *per = [[Person alloc] init];
per.name = @"xiaoming";
per.age = ; // 观察per对象的age属性的变化情况
[per addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:NULL]; // kvoTest里面调用了willChangeValueForKey:和didChangeValueForKey
// 所以:不改变age的值,依旧会被监听到
[per kvoTest]; // 这里po per的结果:<Person: 0x608000026960>
// 但是:po object_getClass(per)的结果:NSKVONotifying_Person
// 所以:苹果偷偷重写了派生类的class方法,但用运行时可以查看当前对象所属类型 // 移除KVO
[per removeObserver:self forKeyPath:@"age"];
} /**
KVO监听:当被监听对象的被监听属性变化后调用 @param keyPath 被监听的属性
@param object 被监听的对象
@param change 变化情况
@param context 携带的参数,此示例传入NULL即可
*/
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
NSLog(@"%@对象的%@属性发生变化了%@",object,keyPath,change); // 打印结果:
// <Person: 0x608000032ca0>对象的age属性发生变化了{
// kind = 1;
// new = 18; // 因为age的值根本没有发生变化,所是这里的new还是18
// }
} @end

KVO的底层实现的更多相关文章

  1. KVO的底层实现原理?如何取消系统默认的KVO并手动触发?

    KVO是基于runtime机制实现的 当某个类的属性对象第一次被观察时,系统就会在运行期动态地创建该类的一个派生类(该类的子类),在这个派生类中重写基类中任何被观察属性的setter 方法.派生类在被 ...

  2. Object-C知识点 (三) 单例 蒙版 刷新 KVO底层

    #pragma mark - 单例方法(完整的方法) 系统的单例方法名称 sharedApplication defaultManager standardUserDefaults currentDe ...

  3. KVO底层实现原理,仿写KVO

    这篇文章简单介绍苹果的KVO底层是怎么实现的,自己仿照KVO的底层实现,写一个自己的KVO监听 #pragma mark--KVO底层实现 第一步:新建一个Person类继承NSObject Pers ...

  4. 简单总结一下NotificationCenter、KVC、KVO、Delegate

    将最近总结的最常用的几种设计模式优势与区别自己小结一下,分享给大家. kvo只能用来对属性作出反应,而不会用来对方法或者动作作出反应,是实现一个对象与另外一个对象保持同步的一种方法,能够提供观察的属性 ...

  5. KVO键值观察的具体实现

    1.KVO简介 KVO是Objective-C对观察者设计模式的一种实现,它提供一种机制,指定一个被观察对象(如A类),当对象中的某个属性发生变化的时候,对象就会接收到通知,并作出相应的处理.在MVC ...

  6. KVO讲解

    最近一直在写swift项目,没有时间更新自己的技术博客,以前在博客里面写过KVO的底层原理,今天我们来看一下KVO的整个使用过程和使用场景(附有demo),大约花大家10-15分钟时间,希望大家看完博 ...

  7. 刨根问底KVO原理

    介绍 KVO( NSKeyValueObserving )是一种监测对象属性值变化的观察者模式机制.其特点是无需事先修改被观察者代码,利用 runtime 实现运行中修改某一实例达到目的,保证了未侵入 ...

  8. OC - KVO实现原理

    1.KVO简介 KVO是Objective-C对观察者设计模式的一种实现,它提供一种机制,指定一个被观察对象(如A类),当对象中的某个属性发生变化的时候,对象就会接收到通知,并作出相应的处理.在MVC ...

  9. iOS:KVC和KVO

    来源:  对月流 链接:http://www.jianshu.com/p/f1393d10109d 写在前面: 关于KVC和KVO各种博客多了去了,重新整理下,就当是温习一下吧,也还算是个新手,不对的 ...

随机推荐

  1. [Asp.net 5] DependencyInjection项目代码分析4-微软的实现(1)

    前面俩种实现中,很多内部细节都无法知道,微软的框架也是为了屏蔽具体实现,只让我们关注接口.但是人都是充满好奇的,依赖注入到底是怎么实现的呢? 微软又有怎样的实现呢?下面就为大家一一呈现(说实话,代码真 ...

  2. VB.NET WinForm获取运行程序用户名

    一个程序也许会被多个用户运行,如下: 那在VB.NET的WinForm环境下,怎样获取User Name呢?可从下面的方法: 代码: Public Shared Function GetProcess ...

  3. 发布网站ASP.NET(ASPX)

    Insus.NET平时是怎样发布自己的开发网站,今天分享给大家.或许你的做法就是跟Insus.NET一样. 这篇还是先在VS创建一个站点,然后再部署至IIS中去. 一般情况之下,Insus.NET创建 ...

  4. 【C#】注意用“划算”的方式使用图标

    先解释一下何谓“划算”:假定一个Winform程序包含若干个窗体,每个窗体左上角都要显示图标(即要设置Form.Icon属性),该程序本身也要有个图标(用于在OS资源管理器中显示),所有这些图标都是一 ...

  5. 紫橙绿蓝的jQuery幻灯片切换

      效果展示 http://hovertree.com/texiao/jquery/77/ 看惯了左右切换的幻灯片,何问起向您推荐一个新颖的,旋转切换,通过点击按钮的相应区域可以使幻灯片以旋转的方式来 ...

  6. SQL SERVER与C#中数据类型的对应关系

    对应关系表 SQL Server2000 http://hovertree.com/menu/sqlserver/ C# CodeSmith 数据类型 取值范围 数据类型 取值范围 空值代替值 数据类 ...

  7. 炉石传说 C# 开发笔记(BS模式Demo)

    原来这个项目,一直想做成CS模式的,BS模式对于炉石这样的游戏来说比较困难. 暴雪到现在也只出了 Windows 和 iPad版本的炉石,最大的问题还是在于如何在小屏幕下,实现最佳的客户体验. Win ...

  8. 【Java每日一题】20161025

    20161024问题解析请点击今日问题下方的"[Java每日一题]20161025"查看 package Oct2016; import static java.lang.Math ...

  9. springmvc中RequestMapping的解析

    在研究源码的时候,我们应该从最高层来看,所以我们先看这个接口的定义: package org.springframework.web.servlet; import javax.servlet.htt ...

  10. Swagger接入

    一 简介 Swagger是一个开源项目,用于描述和生成RestAPi的文档.可以帮助开发人员快速了解web服务的功能. 二  接入Swagger流程 1.在所在Module的pom.xml中,添加Sw ...