KVC、KVO概述
- KVC(NSKeyValueCoding) "鍵-值 編碼"是一種間接訪問對象的屬性的機制,在Object2.0之后系統(tǒng)提供了(.)語法來訪問屬性陋守,在此之前我們需要用KVC來訪問。
- KVO(NSKeyValueObserving) "鍵-值 監(jiān)聽"定義了這樣一種機制,當對象屬性值發(fā)生變化的時候我們能收到一個通知倾贰。
注:NSObject類為所有對象提供了一個自動觀測能力的NSKeyValueObserving協(xié)議。你可以根據(jù)具體需求來打開這個監(jiān)聽機制拦惋。KVO是根據(jù)KVC來實現(xiàn)的匆浙。
KVC、KVO的作用
- KVC是一種間接訪問對象屬性的機制厕妖。
獲取屬性值時可以通過valueForKey:的方法首尼,設置屬性值用setValue: forKey:。與此同時KVC還對未定義的屬性值定義了valueForUnderfineKey:,在這里我們需要注意的是KVC中的value都必須是對象叹放。我們可以通過KVC給在m文件中聲明的屬性進行操作饰恕,前提是,我們必須知道這個屬性的屬性名井仰。 - KVO是基于KVC來實現(xiàn)的埋嵌。
系統(tǒng)已經(jīng)為我們提供了相關(guān)的方法,我們只需要注冊這個監(jiān)聽俱恶,就能很好的監(jiān)聽屬性的值發(fā)生改變雹嗦。 - 注冊:下面的方法中范舀,keyPath就是要觀察的屬性值,options提供的是一個枚舉值了罪,就是當鍵值變化是你可以選擇不同的方式锭环,context方便傳輸你需要的數(shù)據(jù)。
-(void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:
(NSKeyValueObservingOptions)options context:(void *)context
- 實現(xiàn)監(jiān)聽方法:其中change里儲存了一些數(shù)據(jù)的變化泊藕,比如變化前的數(shù)據(jù)辅辩,變化后的數(shù)據(jù)。
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:
(NSDictionary<NSString *,id> *)change context:(void *)context
KVC鍵-值 編碼
KVC的實現(xiàn)機制
KVC會順序使用如下技術(shù)
- 檢查是否存在getter方法-<Key>或者setter方法-set<Key>的方法娃圆。
- 如果沒有上述方法玫锋,則檢查是否存在名字為—_<Key>、<Key>的實例變量讼呢。
- 如果仍未找到撩鹿,則調(diào)用valueForUnderfineKey和setValue:forUnderfineKey:方法。這些方法的默認實現(xiàn)都是拋出異常悦屏,我們可以根據(jù)需要重寫他們节沦。
示例代碼
//使用KVC進行屬性的間接訪問
Person *aperson = [[Person alloc]init];
[aperson setValue:@"Duck" forKey:@"name"];//等效于aperson.name = @"Duck";
[aperson setValue:@"male" forKey:@"gender"];//等效于aperson.sex = @"male";
Student *astudent = [[Student alloc]init];
[aperson setValue:astudent forKey:@"student"];//通過鍵值的方式為student屬性賦值
[aperson setValue:@"Leo" forKeyPath:@"student.name"];//鍵路徑:通過這個方法可以對屬性的屬性進行操作。
setValuesForKeysWithDictionary:<#(nonnull NSDictionary<NSString *,id> *)#>
使用這個方法可以將字典里的鍵值快速的賦值給屬性础爬,但是一定要保證屬性的名字和字典的鍵是一致的甫贯。如果字典的鍵多余屬性那么就需要我們在這個屬性的類中實現(xiàn)下面的一個方法。
-(void)setValue:(id)value forUndefinedKey:(NSString *)key
KVO鍵-值 監(jiān)聽者
KVO key - value - observer 觀察者觀察的是屬性是否執(zhí)行了set方法或者是屬性是否使用了KVC賦值看蚜,只要有賦值的動作获搏,都會執(zhí)行KVO的回調(diào)方法。如果賦值沒有通過set方法或者例如(_name = @"新值"失乾,就不會出發(fā)KVO回調(diào)方法,一般KVO崩潰的原因有兩種常熙,第一個原因,被觀察的對象是一個局部變量碱茁,它已經(jīng)被銷毀掉了裸卫。第二個原因,觀察者被釋放掉了纽竣,但是沒有移除監(jiān)聽墓贿。3,注冊的監(jiān)聽沒有移除掉蜓氨,又重新注冊了監(jiān)聽)聋袋。
首先我們需要注冊一個觀察者,示例代碼如下
[self.view addObserver:self forKeyPath:@"backgroundColor" options:
NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:nil];
options幾個枚舉選項的意思
NSKeyValueObservingOptionOld 把更改之前的值提供給處理方法
NSKeyValueObservingOptionNew 把更改之后的值提供給處理方法
NSKeyValueObservingOptionInitial 把初始化的值提供給處理方法穴吹,一旦注冊幽勒,立馬就會調(diào)用一次。通常它會帶有新值港令,而不會帶有舊值啥容。
NSKeyValueObservingOptionPrior 分2次調(diào)用锈颗。在值改變之前和值改變之后。
- observer:觀察者(觀察對象屬性的變化)
- KeyPath:被觀察屬性的名稱
- options:觀察屬性的新值舊值等一些的配置
- context:可以為KVO的回調(diào)方法傳值咪惠。
KVO的回調(diào)方法
當同時監(jiān)聽多個屬性的時候击吱,我們需要判斷
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context{
if ([keyPath isEqualToString:@"backgroundColor"]) {//view的背景顏色發(fā)生了變化
id oldColour = [change objectForKey:NSKeyValueChangeOldKey];
id newcolour = [change objectForKey:NSKeyValueChangeNewKey];
NSLog(@"%@,old==%@",NSKeyValueChangeOldKey,oldColour);
NSLog(@"%@,new==%@",NSKeyValueChangeNewKey,newcolour);
}
if ([keyPath isEqualToString:@"name"]) {
id oldValue = [change objectForKey:NSKeyValueChangeOldKey];
id newValue = [change objectForKey:NSKeyValueChangeNewKey];
NSLog(@"%@",oldValue);
NSLog(@"%@",newValue);
}
}
- keyPath:屬性名稱
- object:被觀察的對象
- change:變化前后的值都存儲在字典中
- context:注冊觀察者的時候context傳過來的值
移除監(jiān)聽的方法
一定要在我們不需要監(jiān)聽的時候,移除監(jiān)聽遥昧,不然可能會造成程序崩潰覆醇。
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:
(nullable void *)context NS_AVAILABLE(10_7, 5_0);
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;
至此有關(guān)于KVC和KVO的相關(guān)知識解釋完成,還有其他不足之處炭臭,歡迎提出意見叫乌。