KVC和KVO总结

引言

今天看了KVC和KVO的视频,总结一下,便于以后回顾。
KVC地址
KVO地址

KVC(key-value-coding,即键值编码)

KVC 应用场景

  1. 对私有变量进行赋值
  2. 字典转模型

对变量进行赋值和取值

比如有个Person类,有name 和 age 属性,还有个dog属性,dog有体重属性。

Person *p = [[Person alloc] init];
[p setValue:@"张三" forKey: @"name"];
[p setValue:@20 forKey: @"age"];


p.dog = [[Dog alloc] init];
 // 会报错
[p steValue: @200 forKey: @"dog.weight"]; 

// 正确,说明forKeyPath是包含了forKey这个方法的功能的,甚至forKeyPath方法还有它自己的高级的功能,它会先去找有没有dog这个key,然后去找有没有weight这个属性。所以我们在使用kvc的时候,最好用forKeyPath这个方法。
[p setValue: @200 forKeyPath: @"dog.weight"]; 

如果Person还有个 _height的私有属性,如下

#import "Person.h"
@implementation Person
{
  NSInteger _height;
}

- (void)logHeight
{    
  NSLog(@"%ld",_height);
}
@end

这时候是没有办法直接给身高赋值,外满我们访问不到,但是通过KVC就可以了。

[p setValue:@170 forKey:@"height"];

这里我们传入的字符串key是 height,但是定义的属性是 _height,但是通过KVC还是可以给 _height 属性赋值。这说明对某一属性进行赋值,可以不用加下划线,它的查找规则应该是先查找和直接写入的字符串相同的成员变量,如果找不到就找以下划线开头的成员变量。

KVC字典转模型

Person 类对外提供一个接口,将转模型的工作放到模型中。外界可以直接将字典传入,和平时转模型相比,KVC更加方便。

- (instancetype)initWithDict:(NSDictionary *)dict
{
   if (self = [super init]) {
        [self setValuesForKeysWithDictionary:dict];
    }
    returnself;
}

NSDictionary*PersonDict = @{@"name":@"李四",@"age":@"18"};
Person *p = [Person  personWithDict:PersonDict];

KVO(key-value-observing)

KVO用法

利用一个key来找到某个属性并监听其值的改变,是一种典型的观察者模式。
其用法如下:

  • 添加观察者
  • 在观察者中实现监听方法,observeValueForKeyPath:ofObject:change:context:
  • 移除观察者
//让对象b监听对象a的name属性,KVO监听的是set方法
//options属性可以选择是哪个
 /*
     NSKeyValueObservingOptionNew =
0x01, 新值
     NSKeyValueObservingOptionOld =
0x02, 旧值
     */
    [a addObserver:b forKeyPath:@"name"options:kNilOptionscontext:nil];
  a.name = @"zzz";

#pragma mark - 实现KVO回调方法
/*
 *  当对象的属性发生改变会调用该方法
 *  @param keyPath 监听的属性
 *  @param object  监听的对象
 *  @param change  新值和旧值
 *  @param context 额外的数据
 */

- (void)observeValueForKeyPath:(NSString *)keyPath
ofObject:(id)object
change:(NSDictionary<NSString *,id>
*)change context:(void *)context
{
    NSLog(@"%@的值改变了,",keyPath);
    NSLog(@"change:%@", change);
}

// 最后不要忘记,和通知一样,要在dealloc方法里面移除监听

- (void)dealloc
{
    [a removeObserver:bforKeyPath:@"name"];
}

手动触发KVO

上面介绍的是自动触发KVO,值一旦改变就触发。如果手动触发,则要实现一个方法,如下。默认是返回YES。



除此之外,还要在值改变的地方加上两个方法:


观察容器属性

NSMutableArray *tempArr = [_p mutableArrayValueForKey: @"arr"];
[tempArr addObject:@"obje"];

KVO原理

KVO的底层原理还是涉及到了runtime,所以理解其原理之间还是要了解runtime的机制。
当一个类的属性被观察的时候,发生了三件事:

  1. 系统会通过runtime动态的创建一个该类的派生类;
  2. 并且会在这个类中重写基类被观察的属性的setter方法;
  3. 系统将这个类的isa指针指向了派生类。
    这就实现了给监听的属性赋值时调用的是派生类的setter方法,重写的setter方法会在调用原setter方法前后,通知观察对象值的改变。

根据这个原理可以写一个自己的KVO,如下


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 概念 先来看看概念,Key-value coding (KVC) 和 key-value observing (K...
    wuwy阅读 1,422评论 0 1
  • 写在前面:  关于KVC和KVO各种博客多了去了,重新整理下,就当是温习一下吧,不对的地方请指教,喜欢的点个喜欢什...
    shikaiming阅读 23,205评论 21 133
  • KVC kvc全称key-value-coding(键值编码),通常是用来给某一个对象的属性进行赋值,比如有一个p...
    珍此良辰阅读 356评论 0 3
  • KVC(Key-Value Coding) KVC最常见的两个用途: 给私有的成员变量赋值(当然公有的也可以)比如...
    ForeverYoung21阅读 1,185评论 1 15
  • 1.给私有的成员变量赋值(当然公有的也可以) 比如一个类有一个私有成员变量(在.m文件中),用KVC的方式然后可以...
    sunny12阅读 428评论 0 1