Objective-C 学习笔记 - 第18章 键值编程

本章着重介绍键值编程、一系列语言机制和API。Objective-C 的键值编程特性统称为键值编码( Key-Value Coding KVC )和键值观察( Key-Value Observing KVO )。使用键值编码可以通过名称(键)间接访问和操作对象的属性,而无须使用访问方法或支持实例变量。通过键值观察能够使对象在其他对象的属性发生更改时获得通知。

<h3 id="kvc">键值编码</h3>

键值编码提供了一种用于访问对象属性的键值对机制,其中键是属性的名称,而值就是属性的值。

// 代码清单-18.1
//
@interface Hello : NSObject
@property (nonatomic, retain) NSString * greeting;
...
@end

// 使用标准的属性访问方法属性或使用点语法访问属性
[helloObject greeting];
[helloObject setGreeting:newValue];
helloObject.greeting;
helloObject.greeting = newValue;

// 使用键值编码机制访问属性
[helloObject valueForKey:@"greeting"];
[helloObject setValue:@"Hello" forKey:@"greeting"];

通过键值编码可以使用能够在运行时改变的字条串访问属性,从而更加动态和灵活地访问和操作对象的状态。下面是键值编码的几个重要优点。

  • 基于配置的属性访问。通过 KVC 可以使用由参数驱动的通用API访问属性。
  • 降低耦合性。通过 KVC 访问属性可以减少各个软件之间的耦合性,从而提高软件的可维护性。
  • 简化代码。通过 KVC 可以�减少代码量,在需要根据变量访问指定属性时尤为如此。无须使用条件表达式判断需要调用哪个属性访问方法,而直接使用 KVC 表达式,并将变量作为其参数。

如果需要根据用户的输入数据动态更新模型的状态,可以使用标准属性访问方法,也可以使用 KVC 键值编码。

// 代码清单-18.2
//
// 标准属性访问方法
- (void) updateModel:(NSString *)value forState:(NSString *)state {
    if ([state isEqualToString:@"species"]) {
        [self setSpecies:value];
    } else if ([state isEqualToString:@"genus"]) {
        [self setGenus:value];
    } 
    ...
}

// KVC 键值编码
- (void) updateModel:(NSString *)value forState:(NSString *)state {
    [self setValue:value forKey:state];
    ...
}

<h3 id="kvcpath">键和键路径</h3>

键值编码使用键和键路径访问属性。键是用于标识属性的字条串。键路径指明了需要遍历的对象属性序列。

// 代码清单-18.3
//
// 名称类
@interface Name : NSObject
@property (nonatomic, retain) NSString * firstName;
@property (nonatomic, retain) NSString * lastName;
...
@end

// 地址类
@interface Address : NSObject
@property (nonatomic, retain) NSString * street;
@property (nonatomic, retain) NSString * city;
@property (nonatomic, retain) NSString * state;
@property (nonatomic, retain) NSString * zip;
...
@end

// 人类
@interface Person : NSObject
@property (nonatomic, retain) Name * name;
@property (nonatomic, retain) Name * address;
...
@end

// 标准属性访问方法和 KVC 编码
person.name.firstName = @"Bob";
[person setValue:@"Bob" forKeyPath:@"name.firstName"];
NSString *name = [person valueForKeyPath:@"name.firstName"];

// 使用 KVC 获取多个属性的值
NSArray * personKeys = @[@"name", @"address"];
NSDictionary * personValues = [person dictionaryWithValuesForKeys:personKeys];

// 使用 KVC 设置多个属性的值
Name * tom = [Name new];
Address * home = [Address new];
NSDictionary * personProperties = @[@"name":tom, @"address":home];
[person setValuesForKeysWithDictionary:personProperties];

<h3 id="kvcoperate">键值编码的集合操作符</h3>

键值编码含有一系列操作符,使用它们可以通过键路径点表达式对集合元素执行操作。下面是 KVC 集合操作符专用的键路径格式:

集合键路径.@操作符.属性键路径

这些专用路径会被用作 valueForKeyPath: 方法的参数,来执行集合操作。注意,键路径的各个部分之间是用点号分隔的。

  • 集合键路径:如果存在,是指被执行操作的数组或集合(相对于接收对象而言)的键路径。
  • 操作符:带有一个@前缀,是对集合执行的操作。
  • 属性键路径:是操作符所使用集合的属性的键路径。

将 OrderItem 实例的集合存储在名为 ordeItems 的 NSArray 对象后,下面的表达式用带集合操作符的 KVC valueForKeyPath: 表达式。

// 代码清单-18.4
//
@interface OrderItem : NSObject
@property NSString * description;
@property NSString * quantity;
@property flaot * price;
...
@end 

// 计算 OrderItem 实例集合的 price 属性值总和
NSNumber * totalPrice = [orderItems valueForKeyPath:@"@sum.price"];
// 计算 OrderItem 实例集合的含有的对象数量
NSNumber * totalItems = [orderItems valueForKeyPath:@"@count"];

创建一个名为 order 的 Order 实例,使用 @count 操作符可以确定键路径集合中的对象数量。

// 代码清单-18.5
//
@interface Order : NSObject
@property NSArray * items
...
@end 

NSNumber * totalItems = [order valueForKeyPath:@"items@count"];

<h3 id="kvo">键值观察</h3>

键值观察(KVO)是一种通知机制,它使对象能够在其它对象的属性发生更改时获得通知。实际上,它是对观察软件设计模式的实现。它也是模型-视图-控制器(MVC)模式的关键组件。

键值观察的优点非常多,其中包括分割观察对象与被观察对象、提供框架级支持和功能齐全的 API 集合。

// 代码清单-18.6
//
// 通过调用addObserver:forKeyPath:options:context:方法,
// 在观察对象和被观察对象之间建立联系。admin 是观察者,person 是被观察者
// 添加观察对象
Administrator *admin = [Administrator new];
[person addObserver:admin
         forKeyPath:@"name"
            options:NSKeyValueObservingOptionNew
            context:NULL];
            
// 删除观察对象
[person removeObserver:admin forKeyPath:@"name"];

// 当被观察属性的值发生改变时,被观察对象就会调用观察对象中的
// observeValueForKeyPath:ofObject:change:context:方法,在该方法中
// 观察者类实现了用于处理被观察属性更改情况的逻辑。
@implementation Administrator
...
- (void)observeValueForKeyPath:(NSString *)keyPath
                  ofObject:(id)object
                    change:(NSDictionary *)change
                   context:(void *)context {
   if ([@"name" isEqual:keyPath]) {
    // 插入更新 name 属性的逻辑
   } else {
    [super observeValueForKeyPath:keyPath 
                         ofObject:object 
                           change:change 
                          context:context];
   }
}...
@end

<h3 id="kvonotification">键值观察和通知</h3>

通知 NSNotification 实例能够封装通用信息,因此它可以为广泛的系统事件(包括属性更改)提供支持;而键值观察仅支持对象属性更改通知功能,因此在处理纯属性更改情况时,与通知 API 相比,KVO API 会更加简单。

通知类使用交互的广播模型,其中的信息(封装在 NSNotification 对象中)会通过集中式通知中心(NSNotificationCenter实例)分发。从而无须接收对象注册通知功能,即可向一个以上的对象发送消息。通知类既支持同步传递通知,也支持异步传递通知(通过 NSNotificationQueue 实例)。通知机制将通知事件中的发送者和接收者完全隔开,所以二者之间没有直接的双向通信机制,它们必须注册通知才能进行双向通信。通知是由其名称标识的,因此它的名称必须具有唯一性,以确保通知能够被正确的观察者接收到。

键值观察使用点对点的交互模型,因此当属性改变时,被观察对象会直接向已注册的观察者发送通知,而且程序也会一直处于阻塞状态,直到相应的方法执行完为止。

<h3 id="kvckvosummary">小结</h3>

本章介绍了键值编程,它由 Objective-C 语言中一组功能最强大的机制和 API 构成。

  • 通过键值编程可以通过名称(键)间接访问和操作对象的属性,而无须通过访问方法或属性的支持实例变量。
  • 键值编码使用键和键路径访问属性。键用于表示指定属性的字条串。键路径指示了对象的属性序列(在对象图中),用于标识指定属性的遍历路径。KVC API 支持对一个对象中的一个或多个属性进行访问。属性的类型可以为基元、C语言数据结构和对象类型(包括集合)。Objective-C 会使用相应的对象类型,自动封装和解封基元和 C 语言结构。
  • 键值观察可以使对象在其他对象的属性发生更改时获得通知。实际上,它是对观察软件设计模式的实现,也是模型-视图-控制器(MVC)设计模式的关键组件。键值观察以键值编码为基础。
  • 键值观察提供了多个 API ,这些 API 既支持自动属性更改通知功能,也支持手动属性更改通知功能。它们使得观察者在属性更改之前和之后,都能执行逻辑。这些 API 支持特性、一对一和一对多关系属性。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,761评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,953评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,998评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,248评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,130评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,145评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,550评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,236评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,510评论 1 291
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,601评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,376评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,247评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,613评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,911评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,191评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,532评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,739评论 2 335

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,497评论 18 139
  • 什么是键值编码 什么是键值编码 访问实例变量可以通过访问器访问,也可以将属性设置为@public。 与此相对,键值...
    陈_振阅读 459评论 0 1
  • OC的理解与特性 OC作为一门面向对象的语言,自然具有面向对象的语言特性:封装、继承、多态。它既具有静态语言的特性...
    失忆的程序员阅读 471评论 0 1
  • 转自 iOS 面试常见问题最全梳理 序言 目前形势,参加到iOS队伍的人是越来越多,甚至已经到供过于求了。今年,找...
    ZMJun阅读 786评论 0 11
  • THREE.Scene() 场景中的三要素camera 相机spotLight 光源object (plane、c...
    0xmx0阅读 1,521评论 0 0