KVC
綜述
通常,我們使用“.語(yǔ)法”去給對(duì)象賦值,而KVC是使用字符串描述對(duì)象屬性或?qū)傩月窂綇亩鴮?shí)現(xiàn)賦值。NSObject都遵循著屬性的動(dòng)態(tài)讀寫協(xié)議NSKeyValueCoding以支持KVC,它提供了兩個(gè)核心方法:
- (void)setValue:(id)value forKey:(NSString *)key;
- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
前者直接通過(guò)屬性名賦值;后者通過(guò)屬性路徑,如object.id賦值。其底層的執(zhí)行機(jī)制為:先調(diào)用setter方法(即set<Key>),若沒(méi)有找到,則調(diào)用
- (BOOL)accessInstanceVariablesDirectly;
方法,默認(rèn)返回YES,則繼續(xù)尋找key的系列屬性(如_key,isKey等),依然不存在,調(diào)用
- (void)setValue:(id)value forUndefinedKey:(NSString *)key;
此述兩個(gè)方法都可在對(duì)象中重寫,用于處理key不存在的情況。
而對(duì)于在賦值時(shí)不小心傳遞了空值nil時(shí)的異常處理方式,KVC則會(huì)調(diào)用
- (void)setNilValueForKey:(NSString*)key;
重寫此方法可以處理設(shè)置空值的異常。
對(duì)于字典對(duì)象來(lái)說(shuō),valueForKey和objectForKey表現(xiàn)行為一樣,所以用valueForKeyPath處理多級(jí)嵌套的字典會(huì)十分方便。
應(yīng)用
1、給私有變量或readonly變量賦值。
2、給控件的內(nèi)部屬性賦值(如自定義UITextFiled的clearButton,或placeholder的顏色,UIPageControl的_pageImage)(一般可利用runtime獲取控件的內(nèi)部屬性名,Ivar *ivar = class_getInstanceVariable獲取實(shí)例成員變量)。
3、結(jié)合Runtime,model和字典的轉(zhuǎn)換(setValuesForKeysWithDictionary,class_copyIvarList獲取指定類的Ivar成員列表)(多數(shù)情況下使用MJ)。
KVO
綜述
KVO是一種基于KVC實(shí)現(xiàn)的觀察者模式(還有一種模式是NSNotification)。當(dāng)指定的被觀察的對(duì)象的屬性更改了,KVO會(huì)自動(dòng)或手動(dòng)方式通知觀察者。
注冊(cè)
[self.tableView addObserver:self forKeyPath:@"contentOffset" options:NSKeyValueObservingOptionNew context:nil];
回調(diào)
- (void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void*)context { ?}
移除
[self.tableView removeObserver:self forKeyPath:@"contentOffset"];
NSKeyValueObservingOption
NSKeyValueObservingOptionNew在回調(diào)的change字典中,包含變化后的新值;
NSKeyValueObservingOptionOld回調(diào)的change字典中,包含變化前的舊值;
NSKeyValueObservingOptionInitial只會(huì)回調(diào)一次,一般用于獲取屬性的初始值;
NSKeyValueObservingOptionPrior值更改前后各觸發(fā)一次。
手動(dòng)觸發(fā)KVO
首先重寫+ (BOOL)automaticallyNotifiesObserversForKey:(NSString*)key;,此方法會(huì)在當(dāng)前類對(duì)象(self)addObserver注冊(cè)觀察者的時(shí)候調(diào)用,且當(dāng)前生命周期只調(diào)用一次。需要手動(dòng)回調(diào)的key應(yīng)返回NO,其他key返回super此方法。
然后封裝合適的自定義方法,主動(dòng)調(diào)用willChangeValueForKey和didChangeValueForKey,兩個(gè)方法之間可對(duì)當(dāng)前key屬性做自定義賦值操作。(可在重寫key的setter方法操作,也可自定義賦值方法-(void)yourSetMethod:(id)obj;,在需要通知的地方回調(diào)。)
最后,在observeValueForKeyPath回調(diào)中處理通知即可。
然而對(duì)于以上使用KVO的使用需要在不同的地方寫三段代碼這顯然不夠優(yōu)雅。在此推薦一個(gè)facebook出品的優(yōu)秀KVO解決方案,KVOController。用法也十分便捷:
FBKVOController *kvoController = [FBKVOController controllerWithObserver:self];[kvoController observe:object keyPath:@"property" options:NSKeyValueObservingOptionInitial | NSKeyValueObservingOptionNew block:^(id observer, id object, NSDictionary *change) { ?
? ? //handle changes?
? ? [self setNeedsLayout];
}];