1.KVO简介

  KVO是Objective-C对观察者设计模式的一种实现,它提供一种机制,指定一个被观察对象(如A类),当对象中的某个属性发生变化的时候,对象就会接收到通知,并作出相应的处理。在MVC设计架构下的项目,KVO机制很适合实现mode模型和view视图之间的通讯。例如:代码中,在模型类A创建属性数据,在控制器中创建观察者,一旦属性数据发生改变就收到观察者收到通知,通过KVO再在控制器使用回调方法处理实现视图B的更新;

2.实现原理

  KVO的实现依赖于Objective-C强大的runtime,KVO的底层实现是监听setter方法。当观察某对象A时,KVO动态机制会动态创建一个A类的子类,并为这个新的子类重写父类的的setter方法。setter方法随后负责通知观察对象属性的变化。

3.深入理解

  Apple使用了isa混写(isa-swizzling)来实现KVO,当观察对象A的时候(也就是调用对象A注册观察者的方法的时候),KVO机制会动态创建一个A的新的子类:NSKVONotifying_A的新,该类继承自对象A的本类。且KVO会为NSKVONotifying_A重写其父类的setter方法,setter方法会负责在调用原setter方法之前或之后,通知所有观察对象属性值的更改状况。

addobserver方法内部实现:

  在这个方法中,被观察对象的isa指针从指向原来的A类,被KVO机制修改为指向A类的子类-NSKVONotifying_A,来实现当前类属性值改变的监听。

  isa指针的作用:每个对象都有一个isa指针,指向该对象的类。它告诉Runtime系统这个对象的类是什么,所以对象注册为观察者的是时候,isa指针会指向新的子类,那么这个对象就会变成新的子类的对象了。因此,该对象调用setter就会调用已经重写的setter了,从而激活键值通知机制。

子类setter方法剖析:

  KVO的键值观察通知依赖于NSObject的两个方法:willChangeValueForKey didChangeValueForKey,在存取值的前后分别调用的方法。被观察属性发生改变之前,willChangeValueForKey:被调用,通知系统该 keyPath 的属性值即将变更;当改变发生后, didChangeValueForKey: 被调用,通知系统该 keyPath 的属性值已经变更;之后, observeValueForKey:ofObject:change:context: 也会被调用。且重写观察属性的setter 方法这种继承方式的注入是在运行时而不是编译时实现的。

4.自己实现KVO

首先创建一个NSObject的类扩展,.h文件中自定一个一个方法,由于目标是自定义实现KVO,所以只需要在系统添加观察者的方法名前面添加一个前缀,参数值不变。方法实现如下

- (void)CC_addObserver:(NSObject*)observer forKeyPath:(NSString*)keyPath options:(NSKeyValueObservingOptions)options context:(nullablevoid*)context{

/*

1.自定义一个NSCCKVO_XXX子类

2.重写父类的setter,在内部恢复子类的做法,通知观察者

3.修改self的指针,指向新创建的NSCCKVO_XXX子类

*/

//动态生成一个类

NSString*oldClassName =NSStringFromClass([selfclass]);

NSString*newClassName = [@"NSCCKVO_"stringByAppendingString:oldClassName];

constchar* name= [newClassNameUTF8String];

//定义一个类

//参数1:继承的那个类参数2:类名称

Class ccClass=objc_allocateClassPair([selfclass], name,0);

//子类添加setter方法,以setName为例

class_addMethod(ccClass,@selector(setName:), (IMP)setName,"v@:@");

//注册这个类

objc_registerClassPair(ccClass);

//修改self的isa指针

object_setClass(self, ccClass);

//绑定observer到self对象中,将观察者绑定当前对象

objc_setAssociatedObject(self, (__bridgeconstvoid*)@"objc", observer,OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}

voidsetName(idself,SEL_cmd,NSString*newName){

//调用父类的sett方法

Class superClass =class_getSuperclass([selfclass]);

//改变isa指针。为父类,调用set方法

object_setClass(self, superClass);

objc_msgSend(self,@selector(setName:),newName);

//拿出观察者

idobserver =objc_getAssociatedObject(self, (__bridgeconstvoid*)@"objc");

//通知观察者

objc_msgSend(observer,@selector(observeValueForKeyPath:ofObject:change:context:),@"name",self,@{@"new":newName},nil);

//改回子类类型

object_setClass(self, [selfclass]);

}

然后在ViewController导入NSObject的类扩展的头文件

Person*person = [[Personalloc]init];

_person= person;

//[person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew context:nil];

// 这个方法就是刚才自己去实现的那个方法[personCC_addObserver:selfforKeyPath:@"name"options:NSKeyValueObservingOptionNewcontext:nil];

-(void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void*)context{

NSLog(@"%@的%@值变成了%@",object,keyPath,change[@"new"]);

}

此处,自己实现KVO就完成了。观察者观察的是属性,只有遵循 KVO 变更属性值的方式才会执行KVO的回调方法,例如是否执行了setter方法、或者是否使用了KVC赋值。如果赋值没有通过setter方法或者KVC,而是直接修改属性对应的成员变量,例如:仅调用_name = @"newName",这时是不会触发kvo机制,更加不会调用回调方法的。所以使用KVO机制的前提是遵循 KVO 的属性设置方式来变更属性值。

OC - KVO实现原理的更多相关文章

  1. iOS KVO的原理

    KVO(Key Value Observing),是观察者模式在Foundation中的实现.   KVO的原理   简而言之就是:   1.当一个object有观察者时,动态创建这个object的类 ...

  2. KVC, KVO 实现原理

    Key-Value Coding: 键值编码 (KVC) 方法调用: // 对象属性 // 类似: Person -> name setValue: forKey: // 对象的属性或者 属性的 ...

  3. 使用Runtime自定义KVO,原理浅析

    一.介绍 什么是KVO?全称key-value-observer,键值观察,观察者设计模式的另一种实现.其作用是通过观察者监听属性值的变化而做出函数回调. 二.原理 KVO基于Runtime机制实现, ...

  4. KVC, KVO实现原理剖析

    iPhone程序开发 KVO/KVC实现机理分析是本文要介绍的内容,不多说,直接进入话题.我们来看详细内容. Objective-C里面的Key-Value Observing (KVO)机制,非常不 ...

  5. KVO 其原理探究

    什么是KVO ? KVO这是Key-Value Observing,它提供了一种机制,指定的对象的属性被改动后,则对象就会接受到通知. 简单的说就是每次指定的被观察的对象的属性被改动后.KVO就会自己 ...

  6. KVO实现原理

    KVO意为键值观察Key-Value-Observer,本质仍然是观察者模式. 观察者模式的定义:一个目标对象管理所有依赖于它的观察者对象,并在它自身的状态改变时主动通知观察者对象.这个主动通知通常是 ...

  7. OC SEL (@selector) 原理及使用总结(转)

    SEL 类成员方法的指针 可以理解 @selector()就是取类方法的编号,他的行为基本可以等同C语言的中函数指针,只不过C语言中,可以把函数名直接赋给一个函数指针,而Object-C的类不能直接应 ...

  8. 42 (OC)* 字典实现原理--哈希原理

    一.NSDictionary使用原理 1.NSDictionary(字典)是使用 hash表来实现key和value之间的映射和存储的,hash函数设计的好坏影响着数据的查找访问效率. - (void ...

  9. OC - 17.AFNetworking原理及常用操作

    AFN的六大模块 NSURLConnection,主要对NSURLConnection进行了进一步的封装,包含以下核心的类: AFURLConnectionOperation AFHTTPReques ...

随机推荐

  1. C#基础拾遗系列之二:使用ILSpy探索C#7.0新增功能点

    C#基础拾遗系列之二:使用ILSpy探索C#7.0新增功能点   第一部分: C#是一种通用的,类型安全的,面向对象的编程语言.有如下特点: (1)面向对象:c# 是面向对象的范例的一个丰富实现, 它 ...

  2. 面向对象之property

    property功能 以调用数据属性的方式(不用加括号)调用方法 方法定义成数据属性(方法本应该是动词) # 定义property之前 class People: def __init__(self, ...

  3. java自增(自减)运算符

    自增(自减)运算符: ++ --就是可以将当前变量自我增加(减少)1 的运算符. i++, 后++, 先将 i 的值作为整个表达的值, 然后将 i 增加 1. ++i, 先++, 先将 i 增加 ...

  4. SQLSERVER 中GO的作用

    go 向 SQL Server 实用工具发出一批 Transact-SQL 语句结束的信号.go是把t-sql语句分批次执行.(一步成功了才会执行下一步,即一步一个go) BEGIN 和 END 语句 ...

  5. hexo安装-nodejs

    npm- 安装淘宝源 http://npm.taobao.org/$ npm install -g cnpm --registry=https://registry.npm.taobao.org- 查 ...

  6. 用jquery实现的简单数据双向绑定

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. 分布式链路跟踪系统架构SkyWalking和zipkin和pinpoint

    Net和Java基于zipkin的全链路追踪 https://www.cnblogs.com/zhangs1986/p/8966051.html 在各大厂分布式链路跟踪系统架构对比 中已经介绍了几大框 ...

  8. MATLAB入门学习(一)

    开始MATLAB入门啦,,,首先感谢xyy大神的帮助!然后我们开始学习吧!<( ̄︶ ̄)↗[GO!] 工作空间窗口:保存了你定义的常量,变量之类的,可以保存也可以被调用. 保存的话会生成一个mat ...

  9. codeforces 792C. Divide by Three

    题目链接:codeforces 792C. Divide by Three 今天队友翻了个大神的代码来问,我又想了遍这题,感觉很好,这代码除了有点长,思路还是清晰易懂,我就加点注释存一下...分类吧. ...

  10. ADF系列-1.EO的各个属性初探

    ADF中的EO可以看做关系型数据库中表的java展现形式. 1.Type 此处的Type是java的类型,而需要映射的是数据库的类型,两者之间存在一些差异,各个映射关系如下: 数据库列类型 BC 属性 ...