详解KVO 2 底层概念,手动管理KVO

http://www.jianshu.com/p/fe45cdaca4a6

NSKeyValueObserving

KVO的优点:当有属性改变,KVO会提供自动的消息通知。这样开发人员不需要自己去实现这样的方案:每次属性改变了就发送消息通知。这是KVO机制提供的最大的优点。其次,KVO的架构非常的强大,可以很容易的支持多个观察者观察同 一个属性,以及相关的值。
KVC的实现分析KVC运用了一个isa-swizzling技术。
isa-swizzling就是类型混合指针机制。KVC主要通过isa-swizzling,来实现其内部查找定位的。isa指针,就是is a kind of的意思,指向维护分发表的对象的类。该分发表实际上包含了指向实现类中的方法的指针,和其它数据。如下KVC的代码:

[person setValue:@"personName" forKey:@"name"];

就会被编译器处理成:

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

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

KVC在调用方法setValue的时候
(1)首先根据方法名找到运行方法的时候所需要的环境参数。
(2)他会从自己isa指针结合环境参数,找到具体的方法实现的接口。
(3)再直接查找得来的具体的方法实现。
这样的话前面介绍的KVO实现就好理解了
当一个对象注册了一个观察者,被观察对象的isa指针被修改的时候,isa指针就会指向一个中间类,而不是真实的类。
所以isa指针其实不需要指向实例对象真实的类。
所以我们的程序最好不要依赖于isa指针。
在调用类的方法的时候,最好要明确对象实例的类名。
这样只有当我们调用KVC去访问key值的时候KVO才会起作用。所以肯定确定的是,KVO是基于KVC实现的。

KVO简而言之就是:基于键值的观察者,实际上就是观察者模式。
Cocoa Framework已经为我们提供了这一模式,不需要我们自己来实现了。我们只需要按照约定的方式去做就可以了。KVO主要用于用户界面交互,当多个View共同使用了同一个实体,当这个实体中的某个属性改变时,如果需要更新多个界面,KVO的作用就发挥出来了。

有两种方式可以在键值改变的时候给观察者发送通知:自动方式和手动方式。
其中自动方式是由NSObject提供的一个默认实现,通常情况下,如果你自定义了一个类是从NSObject继承而来,那么该类就已经具有了KVO的自动通知功能,而且不需要额外的编写代码。
如果需要手动控制通知方式,那么需要重写automaticallyNotifiesObserversForKey:方法。
在该方法中如果需要手动控制通知方式,则将automaticallyNotifiesObserversForKey:返回NO,否则返回YES。

+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)theKey 
{
     if ([theKey isEqualToString:@"obj"]) {
         return NO ;
     } 
     else {
         return [super automaticallyNotifiesObserversForKey:key] ;
     }    
 }

手动通知方式的好处在于可以减少不必要的通知,比如你可以首先检测一下该属性值是否发生改变,如果发生改变则通知,否则不通知,代码示例如下:

- (void)setObj:(double)obj {
        if (obj != _obj) {
          [self willChangeValueForKey:@"obj"];
          _obj = obj;
          [self didChangeValueForKey:@"obj"];
        }
}

如果一个单一的操作引发了多个属性值的改变,那么就必须嵌套改变通知。代码示例如下:

- (void)setObj:(double)obj {
     [self willChangeValueForKey:@"obj"];
     [self willChangeValueForKey:@"otherObj"];
     _obj = obj;
     _otherObj = otherObj + 1 ;
    [self didChangeValueForKey:@"otherObj"];
    [self didChangeValueForKey:@"obj"];
}

本文部分内容摘录自
http://www.cnblogs.com/pengyingh/articles/2383629.html



文/TEASON(简书作者)
原文链接:http://www.jianshu.com/p/fe45cdaca4a6
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

你可能感兴趣的:(详解KVO 2 底层概念,手动管理KVO)