1. 概述
ObjC主要基于Smalltalk進(jìn)行設(shè)計(jì), 因此它有很多類似Ruby,Python的動(dòng)態(tài)特性, 例如動(dòng)態(tài)類型,動(dòng)態(tài)加載,動(dòng)態(tài)綁定等. 因此可以O(shè)bjC中可以使用鍵值編碼KVC 和 鍵值監(jiān)聽KVO; 基于觀察者思想:
一個(gè)目標(biāo)對(duì)象 管理所有依賴于它的 觀察者對(duì)象;并在自身的狀態(tài)改變時(shí) 主動(dòng)通知觀察者對(duì)象. 通知通告調(diào)用各觀察著對(duì)象所提供的接口方法實(shí)現(xiàn), 觀察者模式 為了解耦;
1. KVC
C#中可以通過反射讀寫一個(gè)對(duì)象的屬性, 利用字符串的方式去動(dòng)態(tài)控制一個(gè)對(duì)象. 但是對(duì)于ObjC的runtime特性, 我們不需要進(jìn)行任何操作即可進(jìn)行屬性的動(dòng)態(tài)讀寫,
KVC的操作方法有NSKeyValueCoding協(xié)議提供, NSObject遵守了這個(gè)協(xié)議, 所以O(shè)C的對(duì)象都可以使用KVC;
- 動(dòng)態(tài)設(shè)置: setValue: forKey:屬性名 (用于簡(jiǎn)單路徑) / setValue: forKeyPath:屬性路徑(用于復(fù)合路徑,即屬性的某屬性.例如Person有一個(gè)Account類型的屬性焰手,那么person.account就是一個(gè)復(fù)合屬性)其實(shí)就是 屬性鏈?zhǔn)皆L問
- 動(dòng)態(tài)讀取: valueForKey: 屬性名 / valueForKeyPath: 屬性路徑
注意:
- KVC 可以訪問私有變量.
- valueForKey會(huì)自動(dòng)把基本類型轉(zhuǎn)成NSNumber或NSValue中包裝成對(duì)象,同樣,動(dòng)態(tài)設(shè)置setValue: forKey:的屬性也必須先包裝成NSNumber對(duì)象類型才可以.
查找規(guī)律:
- 先檢查是否存在屬性a的set和get方法(BOOL類型屬性的get方法名是is<key>), 沒有就會(huì)搜索_<key>/_set<key>方法.
- 如果還沒有再搜索成員變量_a, 如果仍不存在,就會(huì)搜索成員變量a
- 如果最后仍沒搜索到, 會(huì)根據(jù)設(shè)值還是取值 調(diào)用 setValue:forUndefinedKey:或valueforUndefineKey: 拋出異常.根據(jù)需要重寫它們;
補(bǔ)充
批處理:
KVC可以對(duì)對(duì)象進(jìn)行批量更改,dictionaryWithValuesForKeys:和setValuesForKeysWithDictionary:dict;
數(shù)組的整體操作:
如果向一個(gè)數(shù)組請(qǐng)求一個(gè)key,KVC會(huì)查詢數(shù)組中每個(gè)對(duì)象來查找這個(gè)key,之后會(huì)將結(jié)果打包到一個(gè)新數(shù)組并返回;例:student有很多book, 獲取book的nameNSArray *names = [student valueForKeyPath:@"books.name"];
鍵路徑的運(yùn)算符:
在路徑中,可以引用一下運(yùn)算符@xxxxx來進(jìn)行一些運(yùn)算,例如獲取一組值得平均值,最值或者總數(shù).
- 簡(jiǎn)單運(yùn)算符@avg @count @max @min @sum
- 對(duì)象運(yùn)算符: @distinctUnionOfObjects(去掉重復(fù)) @unionOfObjects(不去重復(fù)),都返回?cái)?shù)組
- Array和Set操作符: 集合中包含集合的情況.
NSNumber *count = [student valueForKeyPath:@"books.@count"];//計(jì)算總數(shù)
NSNumber *sum = [student valueForKeyPath:@"books.@sum.price"];//總和
NSArray *prices = [student valueForKeyPath:@"books.@distinctUnionOfObjects.price"];
KVO
//1. 注冊(cè)監(jiān)聽器
-(void)addObserver:(NSObject *)anObserver forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context
anObserver :監(jiān)聽器對(duì)象
keyPath :監(jiān)聽的屬性
options :決定了當(dāng)屬性改變時(shí)芽唇,要傳遞什么數(shù)據(jù)給監(jiān)聽器
//2.監(jiān)聽器需要實(shí)現(xiàn)監(jiān)聽方法,來處理收到的通知;
-(void) observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object
change:(NSDictionary *)change context:(void *)context
keyPath :監(jiān)聽的屬性
object :誰的屬性改變了
change :屬性改變時(shí)傳遞過來的信息(取決于添加監(jiān)聽器時(shí)的options參數(shù)
//3. 最后移除監(jiān)聽器
-(void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath
注冊(cè)和解除注冊(cè)定義在NSKeyValueObserving協(xié)議中, NSObject,NSArray,NSSet實(shí)現(xiàn)了此協(xié)議;
KVO實(shí)現(xiàn)機(jī)制:
當(dāng)某個(gè)類的對(duì)象第一次被觀察時(shí),系統(tǒng)就會(huì)在 運(yùn)行時(shí)動(dòng)態(tài)的創(chuàng)建該類的一個(gè)派生類, 在這個(gè)派生類中重寫原類中被觀察屬性的setter方法;
派生類在被重寫的setter方法中實(shí)現(xiàn)真正的 通知機(jī)制. 這是基于設(shè)置屬性會(huì)調(diào)用setter方法,而通過重寫就可以獲得KVO需要的通知機(jī)制. (所以,使用KVO要遵循其屬性設(shè)置方式來改變屬性值, 如果僅僅直接修改屬性值,是無法實(shí)現(xiàn)KVO的); (補(bǔ)充: Swift中的屬性觀察器原理相似)
- (void) setAge:(int)theAge
{
// will和did兩個(gè)方法用于通知系統(tǒng)該key的屬性值即將和已經(jīng)變更
[self willChangeValueForKey:@"age"];
age = theAge;
[self didChangeValueForKey:@"age"];
}
//didChangeValueForKey方法會(huì)調(diào)用下面方法,
+(BOOL) automaticallyNotifiesObserversForKey:(NSString *)key {
if ([key isEqualToString:@"age"]) {
return NO;
}
return [super automaticallyNotifiesObserversForKey:key];
}
同時(shí)派生類還重寫了class方法以"欺騙"外部調(diào)用者它就是起初的那個(gè)類.然后系統(tǒng)將這個(gè)對(duì)象的isa指針指向了這個(gè)新誕生的派生類, 之后調(diào)用該對(duì)象的setter就會(huì)調(diào)用重寫后的setter從而激活通知
機(jī)制.當(dāng)然,派生類還重寫了dealloc方法來釋放資源.
總結(jié): KVO的三種方式:
- 使用了KVC情況下:如果有訪問器方法,則運(yùn)行時(shí)會(huì)在訪問器方法中調(diào)用
will/didChangeValueForKey:
方法; 如果沒有訪問器方法,運(yùn)行時(shí)會(huì)在setValue:forKey
方法中調(diào)用will/didChangeValueForKey:
方法; - 有訪問器方法: 運(yùn)行時(shí)會(huì)重寫訪問器方法來調(diào)用will/didChangeValueForKey:;
- 如果沒有使用KVC,且沒有訪問器方法, 可以顯示調(diào)用will/didChangeValueForKey:.就同樣可以使用KVO;