KVC、KVO概述
- KVC(NSKeyValueCoding) "鍵-值 編碼"是一種間接訪問對象的屬性的機制,在Object2.0之后系統提供了(.)語法來訪問屬性,在此之前我們需要用KVC來訪問。
- KVO(NSKeyValueObserving) "鍵-值 監聽"定義了這樣一種機制,當對象屬性值發生變化的時候我們能收到一個通知。
注:NSObject類為所有對象提供了一個自動觀測能力的NSKeyValueObserving協議。你可以根據具體需求來打開這個監聽機制。KVO是根據KVC來實現的。
KVC、KVO的作用
- KVC是一種間接訪問對象屬性的機制。
獲取屬性值時可以通過valueForKey:的方法,設置屬性值用setValue: forKey:。與此同時KVC還對未定義的屬性值定義了valueForUnderfineKey:,在這里我們需要注意的是KVC中的value都必須是對象。我們可以通過KVC給在m文件中聲明的屬性進行操作,前提是,我們必須知道這個屬性的屬性名。 - KVO是基于KVC來實現的。
系統已經為我們提供了相關的方法,我們只需要注冊這個監聽,就能很好的監聽屬性的值發生改變。 - 注冊:下面的方法中,keyPath就是要觀察的屬性值,options提供的是一個枚舉值,就是當鍵值變化是你可以選擇不同的方式,context方便傳輸你需要的數據。
-(void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:
(NSKeyValueObservingOptions)options context:(void *)context
- 實現監聽方法:其中change里儲存了一些數據的變化,比如變化前的數據,變化后的數據。
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:
(NSDictionary<NSString *,id> *)change context:(void *)context
KVC鍵-值 編碼
KVC的實現機制
KVC會順序使用如下技術
- 檢查是否存在getter方法-<Key>或者setter方法-set<Key>的方法。
- 如果沒有上述方法,則檢查是否存在名字為—_<Key>、<Key>的實例變量。
- 如果仍未找到,則調用valueForUnderfineKey和setValue:forUnderfineKey:方法。這些方法的默認實現都是拋出異常,我們可以根據需要重寫他們。
示例代碼
//使用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> *)#>
使用這個方法可以將字典里的鍵值快速的賦值給屬性,但是一定要保證屬性的名字和字典的鍵是一致的。如果字典的鍵多余屬性那么就需要我們在這個屬性的類中實現下面的一個方法。
-(void)setValue:(id)value forUndefinedKey:(NSString *)key
KVO鍵-值 監聽者
KVO key - value - observer 觀察者觀察的是屬性是否執行了set方法或者是屬性是否使用了KVC賦值,只要有賦值的動作,都會執行KVO的回調方法。如果賦值沒有通過set方法或者例如(_name = @"新值",就不會出發KVO回調方法,一般KVO崩潰的原因有兩種,第一個原因,被觀察的對象是一個局部變量,它已經被銷毀掉了。第二個原因,觀察者被釋放掉了,但是沒有移除監聽。3,注冊的監聽沒有移除掉,又重新注冊了監聽)。
首先我們需要注冊一個觀察者,示例代碼如下
[self.view addObserver:self forKeyPath:@"backgroundColor" options:
NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:nil];
options幾個枚舉選項的意思
NSKeyValueObservingOptionOld 把更改之前的值提供給處理方法
NSKeyValueObservingOptionNew 把更改之后的值提供給處理方法
NSKeyValueObservingOptionInitial 把初始化的值提供給處理方法,一旦注冊,立馬就會調用一次。通常它會帶有新值,而不會帶有舊值。
NSKeyValueObservingOptionPrior 分2次調用。在值改變之前和值改變之后。
- observer:觀察者(觀察對象屬性的變化)
- KeyPath:被觀察屬性的名稱
- options:觀察屬性的新值舊值等一些的配置
- context:可以為KVO的回調方法傳值。
KVO的回調方法
當同時監聽多個屬性的時候,我們需要判斷
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context{
if ([keyPath isEqualToString:@"backgroundColor"]) {//view的背景顏色發生了變化
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傳過來的值
移除監聽的方法
一定要在我們不需要監聽的時候,移除監聽,不然可能會造成程序崩潰。
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:
(nullable void *)context NS_AVAILABLE(10_7, 5_0);
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;
至此有關于KVC和KVO的相關知識解釋完成,還有其他不足之處,歡迎提出意見。