iOS开发之基础篇(15)—— KVC、KVO

版本

Xcode 9.1

KVC

1、概述

KVC(Key Value Coding)即键值编码,能简便地动态读写对象属性,其实现方法是使用字符串来描述需要更改的对象属性。
KVC的操作方法由NSKeyValueCoding协议提供,而NSObject遵循了该协议,所以说,OC中几乎所有的对象都支持KVC操作。

2、操作方法

  • 写入操作
    setValue:(nullable id) forKey:(NSString *) 用于简单路径
    setValue:(nullable id) forKeyPath:(NSString *) 用于复合路径
  • 读取操作
    valueForKey:(NSString *) 用于简单路径
    valueForKeyPath:(NSString *) 用于复合路径

所谓简单路径,指访问对象本身的属性;所谓复合路径,指访问对象属性里的对象的属性(比如对象A属性里面包含对象B,对象B有属性name,那么对象A访问对象B的name属性即为复合路径)。

示例:
对象Person里面包含属性name、age及对象Dog。对象Dog里有属性name、age。

Person.h

#import "Dog.h"

@interface Person : NSObject

@property (nonatomic, copy)     NSString    *name;
@property (nonatomic, assign)   NSInteger   age;

@property (nonatomic, retain)   Dog         *dog;

@end

Dog.h

@interface Dog : NSObject

@property (nonatomic, copy)     NSString    *name;
@property (nonatomic, assign)   NSInteger   age;

@end

main.m

    // 实例化一个Person
    Person *person = [[Person alloc] init];
    
    // 简单路径的写入操作
    [person setValue:@"King" forKey:@"name"];
    [person setValue:@23 forKey:@"age"];        // 注

    // 简单路径的写入操作(用NSDictionary批量设置)
//    NSDictionary *dic = @{@"name":@"King", @"age":@23,};
//    [person setValuesForKeysWithDictionary:dic];
    
    // 简单路径的读取操作
    NSLog(@"Person name = %@", [person valueForKey:@"name"]);
    NSLog(@"Person age = %ld", [[person valueForKey:@"age"] integerValue]);
    
    
    // 实例化person里的Dog对象属性,否则为nil不能进行如下操作
    person.dog = [[Dog alloc] init];
    
    // 复杂路径的写入操作
    [person setValue:@"XiaoHei" forKeyPath:@"dog.name"];
    [person setValue:@3 forKeyPath:@"dog.age"];
    
    // 复杂路径的读取操作
    NSLog(@"Dog name = %@", [person valueForKeyPath:@"dog.name"]);
    NSLog(@"Dog age = %ld", [[person valueForKeyPath:@"dog.age"] integerValue]);

注:@3是一种简便写法,相当于[NSNumber numberWithInt:3];又如@[]代表数组,@{}代表NSDictionary。如果直接将一个int赋值给id类型的数据,编译会报错。

输出结果:

3、底层实现

  • 写入操作时(例如setValue: forKey:@"A"),方法内部会做以下操作:
  1. 检查是否存在相应key的setter方法(setA),如存在则调用setter方法;
  2. 如果没有setter方法,就会查找与key相同名称并且带下划线的成员变量(_A),如果有则直接赋值;
  3. 如果没有带下划线的成员变量,则搜索与key同名的成员变量(A),有则直接赋值;
  4. 如果最后仍没找到,则调用setValue: forUndefinedKey:方法。
  • 读取操作时(例如valueForKey:@"A"),方法内部会做以下操作:
  1. 检查是否存在相应key的getter方法(A),如存在则调用getter方法;
  2. 如果没有getter方法,就会查找与key相同名称并且带下划线的成员变量(_A),如果有则直接读取;
  3. 如果没有带下划线的成员变量,则搜索与key同名的成员变量(A),有则直接读取;
  4. 如果最后仍没找到,则调用valueForUndefinedKey:方法。

setValue: forUndefinedKey:和valueForUndefinedKey:方法默认实现都是抛出异常,我们可以根据需要重写它们。

KVO

1、概述

KVO(Key Value Observing)即键值监听,是一种观察者模式,通过对某个对象的某个属性添加监听,当该属性改变时,会调用相应方法。
KVO的操作方法由NSKeyValueObserving协议提供,而NSObject遵循了该协议,所以说,OC中几乎所有的对象都支持KVO操作。

2、操作方法

  • 注册指定key路径的观察者:addObserver: forKeyPath: options: context:
  • 回调监听:observeValueForKeyPath: ofObject: change: context:
  • 删除指定key路径的观察者:removeObserver: forKeyPath:

示例(由KVC示例进化):

#import "ViewController.h"
#import "Person.h"

@interface ViewController () {
    
    Person *person;
}

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 实例化一个Person
    person = [[Person alloc] init];     // 因为要移除监听,所以把person设为全局变量
    
    // 简单路径的写入操作
    [person setValue:@"King" forKey:@"name"];
    [person setValue:@23 forKey:@"age"];        // 注
    
    // 简单路径的读取操作
    NSLog(@"Person name = %@", [person valueForKey:@"name"]);
    NSLog(@"Person age = %ld", [[person valueForKey:@"age"] integerValue]);
    
    
    // 实例化person里的Dog对象属性,否则为nil不能进行如下操作
    person.dog = [[Dog alloc] init];
    
    // 复杂路径的写入操作
    [person setValue:@"XiaoHei" forKeyPath:@"dog.name"];
    [person setValue:@3 forKeyPath:@"dog.age"];
    
    // 复杂路径的读取操作
    NSLog(@"Dog name = %@", [person valueForKeyPath:@"dog.name"]);
    NSLog(@"Dog age = %ld", [[person valueForKeyPath:@"dog.age"] integerValue]);
    
    /* 添加对name的监听(注册观察者)*/
    //第一个参数 observer:观察者 
    //第二个参数 keyPath: 被观察的属性名称
    //第三个参数 options: 观察属性的新值、旧值等的一些配置(枚举值,可以根据需要设置,例如这里可以使用两项)
    //第四个参数 context: 上下文,可以为 KVO 的回调方法传值(例如设定为一个放置数据的字典)
    [person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:nil];
    
    person.name = @"Haha";      // 这句会调用如下变化方法
}


/**
 当name发生变化时调用此方法
 
 @param keyPath 属性名称
 @param object 被观察的对象
 @param change 变化前后的值都存储在 change 字典中
 @param context 注册观察者时,context 传过来的值
 */
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    
    if ([keyPath isEqualToString:@"name"]) {
        NSLog(@"newValue:%@, oldValue:%@", [change objectForKey:@"new"],change[@"old"]);
    }
}


- (void)dealloc {
    
    // 移除监听
    [person removeObserver:self forKeyPath:@"name"];
}


@end

结果:

3、优缺点

  • 优点
  1. 能够提供一种简单的方法实现两个对象间的同步。
  2. 能够对系统对象的状态改变作出响应,而不需要改变内部对象的实现;
  3. 能够提供观察的属性的最新值以及先前值;
  4. 用key paths来观察属性,因此也可以观察嵌套对象;
  5. 监听对象可以是空的,为了防止意外崩溃;
  6. 完成了对观察对象的抽象,因为不需要额外的代码来允许观察值能够被观察。
  • 缺点
  1. 观察的属性(对应key)是字符串类型,纯手打容易出错,且编译器不会警告以及检查;
  2. 对属性重构将导致我们的观察代码不再可用;
  3. 只能监测对象属性,不能对方法或者动作做出反应。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容

  • KVC(Key-value coding)键值编码,单看这个名字可能不太好理解。其实翻译一下就很简单了,就是指iO...
    黑暗中的孤影阅读 49,580评论 74 441
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,107评论 29 470
  • KCV 其实由于ObjC的语言特性,你根部不必进行任何操作就可以进行属性的动态读写,这种方式就是Key Value...
    TYM阅读 1,047评论 0 4
  • KVC简介 我们知道可以通过setter、getter方法来设置和修改对象的属性,也知道如何通过简化的点语法来设置...
    请叫我周小帅阅读 473评论 0 1
  • 翻页功能很常见,通常是这样: 高级翻页:自带预告功能。当鼠标移入翻页箭头时,出现相应页面的内容提要。 这个网站,h...
    黑白之间阅读 204评论 0 1