本章著重介紹鍵值編程、一系列語(yǔ)言機(jī)制和API误甚。Objective-C 的鍵值編程特性統(tǒng)稱為鍵值編碼( Key-Value Coding KVC )和鍵值觀察( Key-Value Observing KVO )。使用鍵值編碼可以通過(guò)名稱(鍵)間接訪問(wèn)和操作對(duì)象的屬性谒亦,而無(wú)須使用訪問(wèn)方法或支持實(shí)例變量彬向。通過(guò)鍵值觀察能夠使對(duì)象在其他對(duì)象的屬性發(fā)生更改時(shí)獲得通知。
<h3 id="kvc">鍵值編碼</h3>
鍵值編碼提供了一種用于訪問(wèn)對(duì)象屬性的鍵值對(duì)機(jī)制屡拨,其中鍵是屬性的名稱只酥,而值就是屬性的值褥实。
// 代碼清單-18.1
//
@interface Hello : NSObject
@property (nonatomic, retain) NSString * greeting;
...
@end
// 使用標(biāo)準(zhǔn)的屬性訪問(wèn)方法屬性或使用點(diǎn)語(yǔ)法訪問(wèn)屬性
[helloObject greeting];
[helloObject setGreeting:newValue];
helloObject.greeting;
helloObject.greeting = newValue;
// 使用鍵值編碼機(jī)制訪問(wèn)屬性
[helloObject valueForKey:@"greeting"];
[helloObject setValue:@"Hello" forKey:@"greeting"];
通過(guò)鍵值編碼可以使用能夠在運(yùn)行時(shí)改變的字條串訪問(wèn)屬性,從而更加動(dòng)態(tài)和靈活地訪問(wèn)和操作對(duì)象的狀態(tài)裂允。下面是鍵值編碼的幾個(gè)重要優(yōu)點(diǎn)损离。
- 基于配置的屬性訪問(wèn)。通過(guò) KVC 可以使用由參數(shù)驅(qū)動(dòng)的通用API訪問(wèn)屬性绝编。
- 降低耦合性僻澎。通過(guò) KVC 訪問(wèn)屬性可以減少各個(gè)軟件之間的耦合性,從而提高軟件的可維護(hù)性十饥。
- 簡(jiǎn)化代碼窟勃。通過(guò) KVC 可以?減少代碼量,在需要根據(jù)變量訪問(wèn)指定屬性時(shí)尤為如此绷跑。無(wú)須使用條件表達(dá)式判斷需要調(diào)用哪個(gè)屬性訪問(wèn)方法拳恋,而直接使用 KVC 表達(dá)式,并將變量作為其參數(shù)砸捏。
如果需要根據(jù)用戶的輸入數(shù)據(jù)動(dòng)態(tài)更新模型的狀態(tài)谬运,可以使用標(biāo)準(zhǔn)屬性訪問(wèn)方法,也可以使用 KVC 鍵值編碼垦藏。
// 代碼清單-18.2
//
// 標(biāo)準(zhǔn)屬性訪問(wèn)方法
- (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>
鍵值編碼使用鍵和鍵路徑訪問(wèn)屬性梆暖。鍵是用于標(biāo)識(shí)屬性的字條串。鍵路徑指明了需要遍歷的對(duì)象屬性序列掂骏。
// 代碼清單-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
// 標(biāo)準(zhǔn)屬性訪問(wèn)方法和 KVC 編碼
person.name.firstName = @"Bob";
[person setValue:@"Bob" forKeyPath:@"name.firstName"];
NSString *name = [person valueForKeyPath:@"name.firstName"];
// 使用 KVC 獲取多個(gè)屬性的值
NSArray * personKeys = @[@"name", @"address"];
NSDictionary * personValues = [person dictionaryWithValuesForKeys:personKeys];
// 使用 KVC 設(shè)置多個(gè)屬性的值
Name * tom = [Name new];
Address * home = [Address new];
NSDictionary * personProperties = @[@"name":tom, @"address":home];
[person setValuesForKeysWithDictionary:personProperties];
<h3 id="kvcoperate">鍵值編碼的集合操作符</h3>
鍵值編碼含有一系列操作符轰驳,使用它們可以通過(guò)鍵路徑點(diǎn)表達(dá)式對(duì)集合元素執(zhí)行操作。下面是 KVC 集合操作符專用的鍵路徑格式:
集合鍵路徑.@操作符.屬性鍵路徑
這些專用路徑會(huì)被用作 valueForKeyPath: 方法的參數(shù)弟灼,來(lái)執(zhí)行集合操作级解。注意,鍵路徑的各個(gè)部分之間是用點(diǎn)號(hào)分隔的。
- 集合鍵路徑:如果存在,是指被執(zhí)行操作的數(shù)組或集合(相對(duì)于接收對(duì)象而言)的鍵路徑屡律。
- 操作符:帶有一個(gè)@前綴,是對(duì)集合執(zhí)行的操作芒划。
- 屬性鍵路徑:是操作符所使用集合的屬性的鍵路徑。
將 OrderItem 實(shí)例的集合存儲(chǔ)在名為 ordeItems 的 NSArray 對(duì)象后欧穴,下面的表達(dá)式用帶集合操作符的 KVC valueForKeyPath: 表達(dá)式民逼。
// 代碼清單-18.4
//
@interface OrderItem : NSObject
@property NSString * description;
@property NSString * quantity;
@property flaot * price;
...
@end
// 計(jì)算 OrderItem 實(shí)例集合的 price 屬性值總和
NSNumber * totalPrice = [orderItems valueForKeyPath:@"@sum.price"];
// 計(jì)算 OrderItem 實(shí)例集合的含有的對(duì)象數(shù)量
NSNumber * totalItems = [orderItems valueForKeyPath:@"@count"];
創(chuàng)建一個(gè)名為 order 的 Order 實(shí)例,使用 @count 操作符可以確定鍵路徑集合中的對(duì)象數(shù)量涮帘。
// 代碼清單-18.5
//
@interface Order : NSObject
@property NSArray * items
...
@end
NSNumber * totalItems = [order valueForKeyPath:@"items@count"];
<h3 id="kvo">鍵值觀察</h3>
鍵值觀察(KVO)是一種通知機(jī)制拼苍,它使對(duì)象能夠在其它對(duì)象的屬性發(fā)生更改時(shí)獲得通知。實(shí)際上调缨,它是對(duì)觀察軟件設(shè)計(jì)模式的實(shí)現(xiàn)映屋。它也是模型-視圖-控制器(MVC)模式的關(guān)鍵組件苟鸯。
鍵值觀察的優(yōu)點(diǎn)非常多,其中包括分割觀察對(duì)象與被觀察對(duì)象棚点、提供框架級(jí)支持和功能齊全的 API 集合。
// 代碼清單-18.6
//
// 通過(guò)調(diào)用addObserver:forKeyPath:options:context:方法湾蔓,
// 在觀察對(duì)象和被觀察對(duì)象之間建立聯(lián)系瘫析。admin 是觀察者,person 是被觀察者
// 添加觀察對(duì)象
Administrator *admin = [Administrator new];
[person addObserver:admin
forKeyPath:@"name"
options:NSKeyValueObservingOptionNew
context:NULL];
// 刪除觀察對(duì)象
[person removeObserver:admin forKeyPath:@"name"];
// 當(dāng)被觀察屬性的值發(fā)生改變時(shí)默责,被觀察對(duì)象就會(huì)調(diào)用觀察對(duì)象中的
// observeValueForKeyPath:ofObject:change:context:方法贬循,在該方法中
// 觀察者類實(shí)現(xiàn)了用于處理被觀察屬性更改情況的邏輯。
@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 實(shí)例能夠封裝通用信息桃序,因此它可以為廣泛的系統(tǒng)事件(包括屬性更改)提供支持杖虾;而鍵值觀察僅支持對(duì)象屬性更改通知功能,因此在處理純屬性更改情況時(shí)媒熊,與通知 API 相比奇适,KVO API 會(huì)更加簡(jiǎn)單。
通知類使用交互的廣播模型芦鳍,其中的信息(封裝在 NSNotification 對(duì)象中)會(huì)通過(guò)集中式通知中心(NSNotificationCenter實(shí)例)分發(fā)嚷往。從而無(wú)須接收對(duì)象注冊(cè)通知功能,即可向一個(gè)以上的對(duì)象發(fā)送消息柠衅。通知類既支持同步傳遞通知皮仁,也支持異步傳遞通知(通過(guò) NSNotificationQueue 實(shí)例)。通知機(jī)制將通知事件中的發(fā)送者和接收者完全隔開(kāi)菲宴,所以二者之間沒(méi)有直接的雙向通信機(jī)制贷祈,它們必須注冊(cè)通知才能進(jìn)行雙向通信。通知是由其名稱標(biāo)識(shí)的喝峦,因此它的名稱必須具有唯一性势誊,以確保通知能夠被正確的觀察者接收到。
鍵值觀察使用點(diǎn)對(duì)點(diǎn)的交互模型愈犹,因此當(dāng)屬性改變時(shí)键科,被觀察對(duì)象會(huì)直接向已注冊(cè)的觀察者發(fā)送通知,而且程序也會(huì)一直處于阻塞狀態(tài)漩怎,直到相應(yīng)的方法執(zhí)行完為止勋颖。
<h3 id="kvckvosummary">小結(jié)</h3>
本章介紹了鍵值編程,它由 Objective-C 語(yǔ)言中一組功能最強(qiáng)大的機(jī)制和 API 構(gòu)成勋锤。
- 通過(guò)鍵值編程可以通過(guò)名稱(鍵)間接訪問(wèn)和操作對(duì)象的屬性饭玲,而無(wú)須通過(guò)訪問(wèn)方法或?qū)傩缘闹С謱?shí)例變量。
- 鍵值編碼使用鍵和鍵路徑訪問(wèn)屬性叁执。鍵用于表示指定屬性的字條串茄厘。鍵路徑指示了對(duì)象的屬性序列(在對(duì)象圖中)矮冬,用于標(biāo)識(shí)指定屬性的遍歷路徑。KVC API 支持對(duì)一個(gè)對(duì)象中的一個(gè)或多個(gè)屬性進(jìn)行訪問(wèn)次哈。屬性的類型可以為基元胎署、C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)和對(duì)象類型(包括集合)。Objective-C 會(huì)使用相應(yīng)的對(duì)象類型窑滞,自動(dòng)封裝和解封基元和 C 語(yǔ)言結(jié)構(gòu)琼牧。
- 鍵值觀察可以使對(duì)象在其他對(duì)象的屬性發(fā)生更改時(shí)獲得通知。實(shí)際上哀卫,它是對(duì)觀察軟件設(shè)計(jì)模式的實(shí)現(xiàn)巨坊,也是模型-視圖-控制器(MVC)設(shè)計(jì)模式的關(guān)鍵組件。鍵值觀察以鍵值編碼為基礎(chǔ)此改。
- 鍵值觀察提供了多個(gè) API 趾撵,這些 API 既支持自動(dòng)屬性更改通知功能,也支持手動(dòng)屬性更改通知功能共啃。它們使得觀察者在屬性更改之前和之后占调,都能執(zhí)行邏輯。這些 API 支持特性勋磕、一對(duì)一和一對(duì)多關(guān)系屬性妈候。