iOS KVC與KVO

joker_king發表於2018-12-19

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方法-或者setter方法-set的方法。
  • 如果沒有上述方法,則檢查是否存在名字為—_、的例項變數。
  • 如果仍未找到,則呼叫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的相關知識解釋完成,還有其他不足之處,歡迎提出意見。

相關文章