Objective-C之KVC、KVO

时间:2021-03-23 21:59:28

1,KVC(键值编码)  Key Value Coding

  1.1在C#中,可以通过字符串反射来获取对象,从而对对象的属性进行读写,Object-C中有同样的实现,通过字符串(属性名词)对对象的属性进行读写。

  1.2KVC的操作方法由NSKeyValueCoding协议来定义,而Object对象实现该协议,所以所有的对象都具有KVC功能

    动态设置:setValue:属性值 forKey:属性名    setValue: 属性值forKeyPath:属性路径

    动态读取:valueForKey:属性名  valueForKeyPath:属性路径

  名词解释:属性名(简单路径)  属性路径(复合路径,比如A对象有一个b属性,那么A.b就是复合属性)

  1.3KVC动态设置规则:(假设对a进行设置)首先查找setA方法,如果没有setA方,则查找_a变量,如果没有_a变量,则查找a变量,如果没有a变量,则调用 setValue: forUndefinedKey:方法。对于方法和属性,不管公有还是私有都可以设置成功

  1.4KVC动态读取规则:(假设对a进行读取)首先读取a方法,如果没有a方法,则查找_a变量,如果没有_a变量,则查找a变量,如果没有a变量,则调用valueForUndefinedKey:方法。对于方法和属性,不管公有还是私有都可以设置成功

  1.5 KVC 与 对象的读写方法有什么区别

    1,可以通过属性的名字直接访问属性,而不是调用的属性的accssor。

    2,减少代码量

  for example:场景一:

    People *people = [peoleArray objectAtIndex:row];

    if ([[column identifier] isEqualToString:@"name"]) {

      return [people name];

       }

    if ([[column identifier] isEqualToString:@"age"]) {

      return [people age];

       }

    利用KVC :  return [people valueForKey:[colum identifier]];

  场景二:可以把一个NSDictionary 的元素直接赋值给对象的属性。

    -(id) initWithDictionary:(NSMutableDictionary*) jsonObject

    {

      if((self = [super init]))

        {

           [self init];

      [self setValuesForKeysWithDictionary:jsonObject];

       }

      return self;

    }

    setValuesForKeysWithDictionary:  把NSDictionary中key对应的对象属性设置上NSDictionary中key对应的value.

    异常处理:(1)如果NSDictionary中key 与对象中属性的名称不一致时,我们需要 setValue: forUndefinedKey: 方法来处理没有匹配上的属性,否则setValuesForKeysWithDictionary:将抛出异常(NSUndefinedKeyException)

         (2)如果NSDictionay 内嵌的结构比较复杂:Products{product1{count:xx, sumPrice:xx}}, product2{} ….},我们需要重写 setValue: forKey: 方法。

    1.   -(void) setValue:(id)value forKey:(NSString *)key
    2.   {
    3.     if([key isEqualToString:@"products"])
    4.     {
    5.       for(NSMutableDictionary *productDict in value)
    6.       {
    7.         Prodcut *product = [[Product alloc] initWithDictionary:prodcutDict];
    8.         [self.products addObject:product];
    9.       }
    10.     }
    11.   }

  场景三: 我们需要把数组里People对象的name名字的首字母大写,调用kvc:

       return  [array  valueForKeyPath:@"name.capitalizedString"]  (这里是valueForKeyPath)

  1.6 KVC集合运算符 

    插入一个知识点:NSNumber 封装的数据类型有 Char,UnsignedChar, Short, UnsignedShort, Int, UnsignedInt, Long, UnsignedLong,  LongLong, UnsignedLongLong, Float, Double, Bool, Integer, UnsingedInteger.

            NSValue 封装的数据类型:CGPoint, CGSize, CGRect, CGAffineTransform, UIEdgeInsets, UIOffset.

  集合运算符是一组特殊的key path,只能传递给 valueForKeyPath: 该方法。

  集合运算符的格式:

    Objective-C之KVC、KVO  

  共支持5种运算:@sum @avg @count @max @min  

2,KVO(键值监听)  Key Value Observing (观察者模式)

  2.1KVO的操作方法由NSKeyValueObserving协议来指定,NSObject对象实现了该协议,所以所有的对象都拥有KVO操作。

  2.2KVO操作的方法如下:

    注册指定key路径的监听器: addObserver: forKeyPath: options: context:

    删除指定key路径的监听器:removeObserver: forKeyPath:、 removeObserver: forKeyPath: context:

    监听器回调方法:  observeValueForKeyPath: ofObject: change: contenxt:

   options的值说明:

    NSKeyVlaueObservingOptionNew

    需要补充。。。。。。。

    change:  参数为NSDictionary , key 为 new 和 old

  2.3KVO 与NSNotificationCenter的区别

    好处:1,代码更简洁,不需要修改被观察的类,永远都是观察者在做事情

    不足:1,如果没有observer 监听 keyPath, removeOberser: forKeyPath: 会crash。

       2,很难定位是谁监听了对象的属性

  2.4 KVO的自动触发与手动触发

    自动触发原理:键值观察通知依赖于NSObject的两个方法:willChangeValueForKey: 和 didChangeVlaueForKey:。在一个被观察属性发生改变之前,调用willChangeValueForKey,改变之后调用didChangeVlaueForKey,然后调用 observerValueForKey: ofObject: change: context:.

   模拟一个被观察属性发生改变场景:

  - (void)setNow:(NSDate *)aDate {
      [self willChangeValueForKey:@"now"]; // 运行时,自动添加方法
      _now = aDate;  
      [self didChangeValueForKey:@"now"];// 运行时,自动添加方法
  }

  实现原理, isa-Swizzing.

    原理解释: 当我们监听一个对象时,一个新类会被创建,新类继承自原来对象的类,并重写了被观察属性的setter方法,(运行时,加入了自动触发属性变更的通知),最后通过isa-Swizzing,把这个对象的isa指针,指向新创建的子类。

  Objective-C之KVC、KVO

     该图来自(ios程序猿)