KVC和KVO


KVC

熟悉oc语法的同学也许都会懂得这么一点:在oc中,类的成员变量或是方法是没有绝对私有的。

私有方法直接通过类实例无法访问,但可以借助oc的“编译运行时”机制,也即“瞎子摸黑”机制(个人理解:只要确定了该类有方法A,管你是私有共 有,我用performSelector函数就能调用你),说到这,也许有同学会自然想起,那私有变量如何去访问呢?貌似以前还真没这样搞过,然而现实是 可以的,只不过一般我们把变量设为类的私有变量后是不希望自己或是其他人再去访问的,不然我只能说你自己又在找贱了。

好吧,言归正传,这篇文章主要是想介绍下oc中的KVC、KVO、KVB的实现机制,当然跟我上面说的那些肯定是有关系的啦。。

1. KVC: key-value coding(键值编码)

它 是一种使用字符串标识符,间接访问对象属性的机制,它是很多技术的基础。主要的方法就两对方法: (setValue:forKeyvalueForKey)、setValue:forKeyPathvalueForKeyPath); 这个东西有 什么作用呢,我先不说原理,先说怎么用,例子如下:

@interface A { NSString* foo; }
 ... // 其它代码
 @end 
@interface B { NSString* bar; A* myA; }
 ... // 其它代码 
@end 
@implementation B 
... // 假设 A 类型的对象 a,B 类型的对象
 b A* a = ...;
 B* b = ...;
 NSString* s1 = [a valueForKey:@"foo"]; // 正确
 NSString* s2 = [b valueForKey:@"bar"]; // 正确
 NSString* s3 = [b valueForKey:@"myA"]; // 正确
 NSString* s4 = [b valueForKeyPath:@"myA.foo"]; // 正确
 NSString* s5 = [b valueForKey:@"myA.foo"]; // 错误
 NSString* s6 = [b valueForKeyPath:@"bar"]; // 正确
 ... 
@end

其实上面说的那两对方法使用上基本是一样的,只是valueForKeyPath的值是一个路径(路径之间以点号 . 分割),比如数据成员就是对象自己,寻值过程就会向下深入下去。

注意,这里的数据成员的名字都是使用的字符串的形式。这种使用方法的最好的用处在于将数据(名字)绑定到一些触发器(尤其是方法调用)上, 例如键值对观察(Key-Value Observing, KVO)等。

上述代码说明了类的成员变量也可以使用基类NSObject的那两对方法去访问,不一定直接通过类实例访问,但是这种方式还是有一定的风险,具体危 险情况请参考这个:

http://www.devbean.info/2011/04/from_cpp_to_objc_20/

然后我再说下原理,是俺Copy过来的,大家观赏下: > KVC运用了一个isa- swizzling技术。isa-swizzling就是类型混合指针机制。KVC主要通过isa- swizzling,来实现其内部查找定位的。 isa指针,如其名称所指,(就是is a kind of的意思),指向维护分发表的对象的类。该分发表实际上包含了指向实现类中的方法的指针,和其它数据。

比如说如下的一行KVC的代码:

[site setValue:@"sitename" forKey:@"name"];

就会被编译器处理成:

SEL sel = sel_get_uid ("setValue:forKey:");
IMP method = objc_msg_lookup (site->isa,sel);
method(site, sel, @"sitename", @"name");

首先介绍两个基本概念:

  • SEL数据类型:它是编译器运行Objective-C里的方法的环境参数。
  • IMP数据 类型:他其实就是一个编译器内部实现时候的函数指针。当Objective-C编译器去处理实现一个方法的时候,就会指向一个IMP对象,这个对象是C语言表述的类型(事实上,在Objec+tive-C的编译器处理的时候,基本上都是C语言的)。

这下KVC内部的实现就很清楚的清楚了:一个对象在调用setValue的时候,

  1. 首先根据方法名找到运行方法的时候所需要的环境参 数。
  2. 他会从自己isa指针结合环境参数,找到具体的方法实现的接口。
  3. 再直接查找得来的具体的方法实现。

KVO: key-value observing(键值监听)与KVB: key-value Binding(键值绑定)

这两个机制是结合起来使用的,分别说下,

Key-Value Observing (简写为KVO):当指定的对象的属性被修改了,允许对象接受到通知的机制。每次指定的被观察对象的属性被修改的时候,KVO都会自动的去通知相应的观察者。

  • KVO的优点 当 有属性改变,KVO会提供自动的消息通知。这样的架构有很多好处。首先,开发人员不需要自己去实现这样的方案:每次属性改变了就发送消息通知。这是KVO 机制提供的最大的优点。因为这个方案已经被明确定义,获得框架级支持,可以方便地采用。开发人员不需要添加任何代码,不需要设计自己的观察者模型,直接可 以在工程里使用。其次,KVO的架构非常的强大,可以很容易的支持多个观察者观察同一个属性,以及相关的值。

  • KVB实现的两个基本方法

    • 为对象添加观察者OBserver addObserver:forKeyPath:options:context:
    • 观察者OBserver收到信息的处理函数 observeValueForKeyPath:ofObject:change:context:

KVO和KVB最明显的使用场景就是在一些界面实时显示行很强的地方,比如股票走向、售票余额等,这种方式免去了我们自己操作通知的麻烦,想到这,我发现当初点金和91市场中下载页面进度的显示也完全可以采用这种方式。

原英文链接请点我