关于KVC和KVO

作者: 游走的Fish | 来源:发表于2018-06-25 14:47 被阅读0次

    前言

    KVC/KVO在日常开发中也是经常会使用到,但是还是回到那句话,往往我们在使用一种技术时,却不知道实现原理,以及会忽略一些使用时需要注意的地方。这篇文章会对KVC/KVO做一些原理性的说明,至于使用方式,大家自行度娘,网上还是挺多了,这里就不浪费ctrl+c了。希望文章对大家有帮助,同时欢迎大家指正、交流。

    这篇主要说说KVO,关于KVC的相关请看这篇文章:iOS开发技巧系列---详解KVC(我告诉你KVC的一切),写的非常详细。

    简介

    KVO:Key-Value Observing(键值观察)

    观察者设计模式的一种实现,另外一种是:通知(不在本文主题内)。KVO 提供一种机制,指定一个被观察对象(例如 A 类),当对象某个属性(例如 A 中的name属性)发生更改时,对象会获得通知,可作出相应处理。

    那么KVO的实现原理是怎样的呢?

    我们需要知道的是KVO是基于runtime机制实现的

    类A的属性propertyA第一次被观察时,系统就会在运行期动态地创建该类的一个派生类:NSKVONotifying_A,在这个派生类中重写A类中被观察属性的setter方法以及classdealloc_isKVOA方法。派生类在被重写的setter方法内实现真正的通知机制。

    每个类对象中都有一个isa指针指向当前类,当类对象第一次被观察,那么系统会将isa指针指向动态生成的派生类,从而在给被监控属性赋值时执行的是派生类的setter方法
    键值观察通知依赖于NSObject的两个方法: willChangeValueForKey:didChangevlueForKey:;在一个被观察属性发生改变之前, willChangeValueForKey:一定会被调用,这就会记录旧的值。而当改变发生后,didChangeValueForKey:会被调用,继而 observeValueForKey:ofObject:change:context:也会被调用。用代码显示就是:

    -(void)setName:(NSString *)newName{ 
      [self willChangeValueForKey:@"name"];    //KVO 在调用存取方法之前总调用 
      [super setValue:newName forKey:@"name"]; //调用父类的存取方法 
      [self didChangeValueForKey:@"name"];     //KVO 在调用存取方法之后总调用
    }
    

    这里需要注意的一个小问题是:观察者观察的是属性,只有遵循 KVO 变更属性值的方式才会执行 KVO 的回调方法,例如是否执行了 setter 方法、或者是否使用了 KVC 赋值。
    如果赋值没有通过 setter 方法或者 KVC,而是直接修改属性对应的成员变量,例如:仅调用_name = @"newName",这时是不会触发 KVO 机制,更加不会调用回调方法的。
    所以使用 KVO 机制的前提是遵循 KVO 的属性设置方式来变更属性值。

    上文提到的动态修改了isa指针,isa 指针的作用:每个对象都有 isa 指针,指向该对象的类,它告诉 Runtime 系统这个对象的类是什么。所以对象注册为观察者时,isa 指针指向新子类,那么这个被观察的对象就神奇地变成新子类的对象(或实例)了。即重写class方法,是为了我们调用它的时候返回跟重写之前同样的内容。

    NSLog(@"self->isa:%@",self->isa);  
    NSLog(@"self class:%@",[self class]);  
    

    在建立KVO监听前,打印结果为:

    self->isa:Person  
    self class:Person  
    

    在建立KVO监听之后,打印结果为:

    self->isa:NSKVONotifying_Person  
    self class:Person
    

    此外派生类还重写了 dealloc 方法来释放资源,具体的笔者没有深究,网上的文章也很少有提及的。

    KVO不得不关注的要点

    大家可以看这篇文章,主要就是:
    1、对同一个keypath进行两次removeObserver时会导致程序crash(多出现在父类也对同一个keypath监听,dealloc中也removeObserver);
    2、不需要监听的时候,要及时remove,否则被监听对象释放后,再触发监听器会引起崩溃
    3、父类也实现了KVO时的注意点,子类监听回调中需要做处理。
    iOS下KVO使用过程中的陷阱

    题外:如何手动触发KVO

    近期同事问我,如何手动触发一个value的KVO?
    如果上述的内容看过之后,相信就比较容易了。只需要重写待监听类的+(BOOL)automaticallyNotifiesObserversForKey:方法,然后重写setter方法,在赋值前后分别加上willChangeValueForKey:didChangeValueForKey即可,当然监听类仍需addObserver。

    总结

    以上就是笔者对KVO的认识了,当然也借鉴了网上的一些博文的知识,如有错误,还请大家评论指出,大家一起学习,谢谢大家!

    相关文章

      网友评论

        本文标题:关于KVC和KVO

        本文链接:https://www.haomeiwen.com/subject/pzweyftx.html