首页 > 代码库 > (知其所以然 主题3)论观察者模式之KVC和KVO

(知其所以然 主题3)论观察者模式之KVC和KVO

    在开发的时候,是不是忽然有种错觉:我们好像是代码的搬运工,一个项目开始,把自己写好的、封装好的类、框架亦或别人写好的第三方框架不假思索的运用到项目中,一方面:项目时间紧;二方面:简单好用,减少了代码量。但是有时候,我们是否应该停下脚步, 从底层去看看代码的世界,那样我们收益会颇丰,让我们一起走在学习的路上吧!

     在开发中,有很多地方都需要对KVC和KVO进行运用。我们只知道这是Object-C提供的一个不错的机制,可以很好的减少代码;但是我们可能不会过多的去了解它底层是如何实现的,下面,我们一起来看看吧!

   (一) KVC的实现与运用

    KVC 的底层实现是运用了 isa-swizzling 技术,也就是类型混合指针机制。通过 isa-swizzling,能够来实现其内部查找定位的。isa 指针,就是 is a kind of 的意思,指向系统维护分发表的对象的类。该分发表实际上包含了指向实现类中的方法的指针和其它数据对象。

   比如下面的这句代码:

1 [Account  setValue:@"userAccount" forKey:@"account"];
2 
3 经过KVC内部编译后,其内部实现如下:
4     
5 SEL sel = sel_get_uid("setValue:forKey:");
6 IMP method = objc_msg_lookup (account->isa,sel);
7 method(account,sel,@"userAccount",@"account");

我们就可以分析得出:

      SEL是一种数据类型,是编译器运行Objective-C里的方法的环境参数。

      IMP也是一种数据类型,说白了就是编译器内部实现时候的函数指针。也就是说:当编译器去处理实现一个方法的时候,就会指向一个IMP对象,这个对象是C语言表述的类型。  

  KVC在调用方法setValue的时候,会依照以下步骤执行:

     a)根据方法名找到运行方法的时候所需要的环境参数;

     b) 结合isa指针,找到具体的方法实现的接口;

     c) 对查找的方法进行具体的实现

   所以,KVC的运作流程就是:当一个对象注册了一个观察者,被观察对象的isa指针被修改的时候,isa指针就会指向一个中间类,而不是真实的类。比如说:我们在运用KVC进行字典和模型之间转化时,要完全一致对象实例的类名,就是这个道理。

(二)KVO的实现和运用

   KVO是指当指定的对象的属性被修改了,让要进行事件处理的对象及时接收到通知的一种实现机制。

1 定义:
2 [self addObserver:self forKeyPath:@"code" options:0 context:str];
3 实现
4 - (void)observeValueForKeyPath:(NSString *)keyPath   ofObject:(id)object   change:(NSDictionary *)change  context:(void *)context 

      在运用KVO的时候,当有属性发生改变时,会提供自动的消息通知。这样我们在开发的时候就不需要去具体实现(运用代理、block),如每次属性改变了就发送消息通知。我们不需要设计自己的观察者模型,直接可以在项目里进行使用,十分快捷和方便。

    此外,KVO 的架构非常的强大,有框架支持,可以很容易的支持多个观察者观察同一个属性,以及相关的值。

     具体的实例会在稍后发表,让我们在案例中进一步体会这两种观察者模式的运用和实现。