iOS基礎之KVC與KVO

1. 概述

ObjC主要基于Smalltalk進行設計, 因此它有很多類似Ruby,Python的動態(tài)特性, 例如動態(tài)類型,動態(tài)加載,動態(tài)綁定等. 因此可以ObjC中可以使用鍵值編碼KVC 和 鍵值監(jiān)聽KVO; 基于觀察者思想:

一個目標對象 管理所有依賴于它的 觀察者對象;并在自身的狀態(tài)改變時 主動通知觀察者對象. 通知通告調用各觀察著對象所提供的接口方法實現(xiàn), 觀察者模式 為了解耦;

1. KVC

C#中可以通過反射讀寫一個對象的屬性, 利用字符串的方式去動態(tài)控制一個對象. 但是對于ObjC的runtime特性, 我們不需要進行任何操作即可進行屬性的動態(tài)讀寫,

KVC的操作方法有NSKeyValueCoding協(xié)議提供, NSObject遵守了這個協(xié)議, 所以OC的對象都可以使用KVC;

  • 動態(tài)設置: setValue: forKey:屬性名 (用于簡單路徑) / setValue: forKeyPath:屬性路徑(用于復合路徑,即屬性的某屬性.例如Person有一個Account類型的屬性,那么person.account就是一個復合屬性)其實就是 屬性鏈式訪問
  • 動態(tài)讀取: valueForKey: 屬性名 / valueForKeyPath: 屬性路徑

注意:

  1. KVC 可以訪問私有變量.
  2. valueForKey會自動把基本類型轉成NSNumber或NSValue中包裝成對象,同樣,動態(tài)設置setValue: forKey:的屬性也必須先包裝成NSNumber對象類型才可以.

查找規(guī)律:

  1. 先檢查是否存在屬性a的set和get方法(BOOL類型屬性的get方法名是is<key>), 沒有就會搜索_<key>/_set<key>方法.
  2. 如果還沒有再搜索成員變量_a, 如果仍不存在,就會搜索成員變量a
  3. 如果最后仍沒搜索到, 會根據(jù)設值還是取值 調用 setValue:forUndefinedKey:或valueforUndefineKey: 拋出異常.根據(jù)需要重寫它們;

補充

批處理:

KVC可以對對象進行批量更改,dictionaryWithValuesForKeys:和setValuesForKeysWithDictionary:dict;

數(shù)組的整體操作:

如果向一個數(shù)組請求一個key,KVC會查詢數(shù)組中每個對象來查找這個key,之后會將結果打包到一個新數(shù)組并返回;例:student有很多book, 獲取book的nameNSArray *names = [student valueForKeyPath:@"books.name"];

鍵路徑的運算符:

在路徑中,可以引用一下運算符@xxxxx來進行一些運算,例如獲取一組值得平均值,最值或者總數(shù).

  1. 簡單運算符@avg @count @max @min @sum
  2. 對象運算符: @distinctUnionOfObjects(去掉重復) @unionOfObjects(不去重復),都返回數(shù)組
  3. Array和Set操作符: 集合中包含集合的情況.
NSNumber *count = [student valueForKeyPath:@"books.@count"];//計算總數(shù)
NSNumber *sum = [student valueForKeyPath:@"books.@sum.price"];//總和
NSArray *prices = [student valueForKeyPath:@"books.@distinctUnionOfObjects.price"]; 

KVO

//1. 注冊監(jiān)聽器
-(void)addObserver:(NSObject *)anObserver forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context
anObserver :監(jiān)聽器對象
keyPath :監(jiān)聽的屬性
options :決定了當屬性改變時,要傳遞什么數(shù)據(jù)給監(jiān)聽器

//2.監(jiān)聽器需要實現(xiàn)監(jiān)聽方法,來處理收到的通知;
-(void) observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object
change:(NSDictionary *)change context:(void *)context
keyPath :監(jiān)聽的屬性
object :誰的屬性改變了
change :屬性改變時傳遞過來的信息(取決于添加監(jiān)聽器時的options參數(shù)

//3. 最后移除監(jiān)聽器
-(void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath

注冊和解除注冊定義在NSKeyValueObserving協(xié)議中, NSObject,NSArray,NSSet實現(xiàn)了此協(xié)議;

KVO實現(xiàn)機制:

當某個類的對象第一次被觀察時,系統(tǒng)就會在 運行時動態(tài)的創(chuàng)建該類的一個派生類, 在這個派生類中重寫原類中被觀察屬性的setter方法;
派生類在被重寫的setter方法中實現(xiàn)真正的 通知機制. 這是基于設置屬性會調用setter方法,而通過重寫就可以獲得KVO需要的通知機制. (所以,使用KVO要遵循其屬性設置方式來改變屬性值, 如果僅僅直接修改屬性值,是無法實現(xiàn)KVO的); (補充: Swift中的屬性觀察器原理相似)

- (void) setAge:(int)theAge
{
    // will和did兩個方法用于通知系統(tǒng)該key的屬性值即將和已經(jīng)變更
    [self willChangeValueForKey:@"age"];
    age = theAge;
    [self didChangeValueForKey:@"age"]; 
}
//didChangeValueForKey方法會調用下面方法,
+(BOOL) automaticallyNotifiesObserversForKey:(NSString *)key {
    if ([key isEqualToString:@"age"]) {
        return NO;
    }
    return [super automaticallyNotifiesObserversForKey:key];
}

同時派生類還重寫了class方法以"欺騙"外部調用者它就是起初的那個類.然后系統(tǒng)將這個對象的isa指針指向了這個新誕生的派生類, 之后調用該對象的setter就會調用重寫后的setter從而激活通知
機制.當然,派生類還重寫了dealloc方法來釋放資源.

總結: KVO的三種方式:
  1. 使用了KVC情況下:如果有訪問器方法,則運行時會在訪問器方法中調用will/didChangeValueForKey:方法; 如果沒有訪問器方法,運行時會在setValue:forKey方法中調用will/didChangeValueForKey:方法;
  2. 有訪問器方法: 運行時會重寫訪問器方法來調用will/didChangeValueForKey:;
  3. 如果沒有使用KVC,且沒有訪問器方法, 可以顯示調用will/didChangeValueForKey:.就同樣可以使用KVO;
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容