美文网首页
OC总结篇 - KVC和KVO

OC总结篇 - KVC和KVO

作者: 亲爱的大倩倩 | 来源:发表于2019-11-29 16:56 被阅读0次
    KVC
    什么是KVC

    是一种键值编码机制(key-value),通过NSKeyValueCoding协议来间接访问成员变量
    它会破坏面向对象编程思想,上面的key是没有任何限制的,当我们知道一个类内部的某个私有成员变量名称,就可以通过key进行设置和访问

    作用

    可以随意修改一个对象的属性或者成员变量,因为key没有限制,私有也可以

    怎么用
    赋值: 设置某一对象当中和key同名或者相似名字的实例变量的值
    - (void)setValue:(id)value forKey:(NSString *)key;
    - (void)setValue:(id)value forKeyPath:(NSString *)key; 通过点可层层赋值
    
    取值: 获取和key同名或者相似名字的实例变量的值
    - (id)valueForKey:(NSString *)key;
    - (id)valueForKeyPath:(NSString *)key;
    
    系统内部实现流程
    `赋值过程`
    先找相关方法set<Key>  _set<Key>  setIs<Key>
    若没有相关方法,判断是否可以直接方法成员变量accessInstanceVariablesDirectly == YES ?
    YES:继续找相关变量  _<Key>  _is<Key>  <Key>  is<Key>赋值,若都不存在,则调forUndefinedKey,抛出异常
    NO:执行setValue: forUndefinedKey 系统抛出异常:未定义的key
    
    
    取值过程
    先找相关方法get<Key>  <Key>  <countOfKey> && objectInKeyAtIndex
    若没有相关方法,判断是否可以直接方法成员变量accessInstanceVariablesDirectly == YES ?
    YES: 继续找相关变量  _<Key>  _is<Key>  <Key>  is<Key>赋值,若都不存在,则调forUndefinedKey,抛出异常
    NO: 执行setValue: forUndefinedKey 系统抛出异常:未定义的key
    
    其他
    赋值为空: setNilValueForKey
    key值不存在: setValue: forUndefinedKey
    validateValue方法的工作原理: 属性值正确性验证,用来检查set的值是否正确、为不正确的值做一个替换值或者拒绝设置新值并返回错误原因
    
    
    
    关于setValue的调用流程
    关于valueForKey的调用流程
    KVO - Key-value observing
    什么是KVO
    1. 是OC对于观察者设计模式的一种实现,每次当被观察对象的某个属性值发生改变时,注册的观察者便能获得通知。
    2. KVO是NSObject的一个分类
    KVO怎么用
    1.添加观察
    [对象 addObserver:观察者 forKeyPath:被监听的属性 options:值如何变化 context:可以传参];
    
    2.观察回调
    [observeValueForKeyPath:哪个属性被改了 ofObject:哪个对象的属性被改了 change:值 context:参数]
    
    3.移除removeObserver
    
    KVO是如何实现的

    Apple使用isa混写技术(isa-swizzling)来实现KVO,本质是重写了setter方法



    当调用了addObserver: forKeyPath: 方法后,系统会在运行时,为这个类动态的创建了一个子类(NSKVONotifying_A),改写isa指向(将原来类A的isa指针指向NSKVONotifying_A这个类),同时重写setter方法,来实现KVO机制
    通过下图可以看到,在调用[addObserver forKeyPath]之前,类还是MObject,在调用方法之后,类就变成了NSKVONotifying_MObject,其中isa指针的指向也发生了修改

    KVO原理是重写setter方法

    NSKVONotifying_A类实际是类A的子类,之所以继承是为了重写类A的setter方法, NSKVONotifying_A通过对setter方法的重写,达到了可以通知所有观察者的目的

    为什么更改setter就可以实现KVO的监听呢

    NSKVONotifying_A重写了原来类的setter方法
    具体实现是下面两行代码

    - (void)willChangeValueForKey:(NSString *)key;
    - (void)didChangeValueForKey:(NSString *)key;
    
    重写setter方法的具体实现

    如图所示
    首先调用willChangeValueForKey方法
    然后调用父类实现,也就是原来类的setter实现
    然后调用didChangeValueForKey,这个方法会触发observeValueForKeyPath这个KVO回调,来通知观察者value发生了变化

    我们做哪些操作可以触发KVO(Demo百度网盘)

    因为本质是重写setter,所以肯定是要和setter相关才可以

    1.setter相关:点方法和KVC赋值(内部是setter赋值)
    点方法: self.p.name = @"111";
    KVC赋值: [self.p setValue:@"111" forKey:@"name"];
    
    2.自己写代码手动调用
    KVO内部也是在setter方法中调用这两句代码
    在值改变前调用方法[self willChangeValueForKey:@"value"];
    在值改变后调用方法[self didChangeValueForKey:@"value"];
    
    哪些操作不响应KVO

    若通过方法调用直接给成员变量赋值,不会被监听

    [obj increase];
    
    - (void)increase
    
    {
    
        _value += 1;
    
    }
    

    总结

    1. 使用setter方法改变值,KVO回调可以生效
    2. 使用setValue: forKey: 改变值,KVO回调可以生效
    3. 成员变量直接赋值KVO无法生效,必须手动添加KVO两个方法,KVO回调才会生效

    相关文章

      网友评论

          本文标题:OC总结篇 - KVC和KVO

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