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. 如何给wordpress首页自动显示文章内容的第一个图片

    敏捷个人手机应用中使用到的数据来源于wordpress中,因为自己写的页面,所以可以自己写代码获取文章内容的第一个图片作为文章缩略图来显示,这样用户看到首页时图文并茂,感觉会好一些. 现在后台简单的使 ...

  2. 【转】FastCgi与PHP-fpm关系

    刚开始对这个问题我也挺纠结的,看了<HTTP权威指南>后,感觉清晰了不少. 首先,CGI是干嘛的?CGI是为了保证web server传递过来的数据是标准格式的,方便CGI程序的编写者. ...

  3. VS2015 新Web项目(C#6)出现CS1617错误的解决

    VS2015新增了对C#6的支持. 在新的Web项目模板中通过引入nuget包Microsoft.CodeDom.Providers.DotNetCompilerPlatform:1.0.0并在web ...

  4. 删除html元素

    如果需要将id是‘div2js’的div元素删除. 1.使用DOM对象 首先需要找到被删元素的父元素,通过父元素将其需要删除的子元素删除. var el = document.getElementBy ...

  5. Rebalance Customer Balances Utility的使用

    用户不管是打开A/R Posted Transactions Detail还是A/R Posted Transactions Summary 窗口,均显示如下一个警示: 打开Currency Code ...

  6. .Net实现拉勾网爬虫

    前几天看到一个.NET Core写成的爬虫,有些莫名的小兴奋,之前一直用集搜客去爬拉勾网的招聘信息,这个傻瓜化工具相当于用HTML模板页去标记DOM节点,然后在浏览器窗口上模拟人的浏览行为同时跟踪节点 ...

  7. ASP.NET Core开发-使用Nancy框架

    Nancy简介 Nancy是一个轻量级的独立的框架,下面是官网的一些介绍: Nancy 是一个轻量级用于构建基于 HTTP 的 Web 服务,基于 .NET 和 Mono 平台,框架的目标是保持尽可能 ...

  8. Java final 修饰符知识点总结

    final从字面上理解含义为“最后的,最终的”.在Java中也同样表示出此种含义. final可以用来修饰变量(包括类属性.对象属性.局部变量和形参).方法(包括类方法和对象方法)和类. 1. fin ...

  9. C# ~ 由 IDisposable 到 GC

    IDisposable 接口 1. 托管资源和非托管资源   ·  托管资源  a.  CLR 控制和管理的内存资源,如程序中在 Heap 上分配的对象.作用域内的变量等:  b.  GC 机制实现自 ...

  10. Android短信Notification的几个ID

    private static final int NOTIFICATION_ID = 123; public static final int MESSAGE_FAILED_NOTIFICATION_ ...