iOS基礎(chǔ)之KVC與KVO

1. 概述

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

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

1. KVC

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

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

  • 動(dòng)態(tài)設(shè)置: setValue: forKey:屬性名 (用于簡(jiǎn)單路徑) / setValue: forKeyPath:屬性路徑(用于復(fù)合路徑,即屬性的某屬性.例如Person有一個(gè)Account類型的屬性焰手,那么person.account就是一個(gè)復(fù)合屬性)其實(shí)就是 屬性鏈?zhǔn)皆L問
  • 動(dòng)態(tài)讀取: valueForKey: 屬性名 / valueForKeyPath: 屬性路徑

注意:

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

查找規(guī)律:

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

補(bǔ)充

批處理:

KVC可以對(duì)對(duì)象進(jìn)行批量更改,dictionaryWithValuesForKeys:和setValuesForKeysWithDictionary:dict;

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

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

鍵路徑的運(yùn)算符:

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

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

KVO

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

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

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

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

KVO實(shí)現(xiàn)機(jī)制:

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

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

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

總結(jié): KVO的三種方式:
  1. 使用了KVC情況下:如果有訪問器方法,則運(yùn)行時(shí)會(huì)在訪問器方法中調(diào)用will/didChangeValueForKey:方法; 如果沒有訪問器方法,運(yùn)行時(shí)會(huì)在setValue:forKey方法中調(diào)用will/didChangeValueForKey:方法;
  2. 有訪問器方法: 運(yùn)行時(shí)會(huì)重寫訪問器方法來調(diào)用will/didChangeValueForKey:;
  3. 如果沒有使用KVC,且沒有訪問器方法, 可以顯示調(diào)用will/didChangeValueForKey:.就同樣可以使用KVO;
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末扫尺,一起剝皮案震驚了整個(gè)濱河市吉执,隨后出現(xiàn)的幾起案子昔榴,更是在濱河造成了極大的恐慌严肪,老刑警劉巖键思,帶你破解...
    沈念sama閱讀 211,376評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異阐枣,居然都是意外死亡马靠,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,126評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門蔼两,熙熙樓的掌柜王于貴愁眉苦臉地迎上來甩鳄,“玉大人,你說我怎么就攤上這事额划∶羁校” “怎么了?”我有些...
    開封第一講書人閱讀 156,966評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵锁孟,是天一觀的道長(zhǎng)彬祖。 經(jīng)常有香客問我茁瘦,道長(zhǎng)品抽,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,432評(píng)論 1 283
  • 正文 為了忘掉前任甜熔,我火速辦了婚禮圆恤,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘腔稀。我一直安慰自己盆昙,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,519評(píng)論 6 385
  • 文/花漫 我一把揭開白布焊虏。 她就那樣靜靜地躺著淡喜,像睡著了一般。 火紅的嫁衣襯著肌膚如雪诵闭。 梳的紋絲不亂的頭發(fā)上炼团,一...
    開封第一講書人閱讀 49,792評(píng)論 1 290
  • 那天澎嚣,我揣著相機(jī)與錄音,去河邊找鬼瘟芝。 笑死易桃,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的锌俱。 我是一名探鬼主播晤郑,決...
    沈念sama閱讀 38,933評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼贸宏!你這毒婦竟也來了造寝?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,701評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤锚赤,失蹤者是張志新(化名)和其女友劉穎匹舞,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體线脚,經(jīng)...
    沈念sama閱讀 44,143評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡赐稽,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,488評(píng)論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了浑侥。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片姊舵。...
    茶點(diǎn)故事閱讀 38,626評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖寓落,靈堂內(nèi)的尸體忽然破棺而出括丁,到底是詐尸還是另有隱情,我是刑警寧澤伶选,帶...
    沈念sama閱讀 34,292評(píng)論 4 329
  • 正文 年R本政府宣布史飞,位于F島的核電站,受9級(jí)特大地震影響仰税,放射性物質(zhì)發(fā)生泄漏构资。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,896評(píng)論 3 313
  • 文/蒙蒙 一陨簇、第九天 我趴在偏房一處隱蔽的房頂上張望吐绵。 院中可真熱鬧,春花似錦河绽、人聲如沸己单。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,742評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽纹笼。三九已至,卻和暖如春苟跪,著一層夾襖步出監(jiān)牢的瞬間廷痘,已是汗流浹背矮嫉。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評(píng)論 1 265
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留牍疏,地道東北人蠢笋。 一個(gè)月前我還...
    沈念sama閱讀 46,324評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像鳞陨,于是被迫代替她去往敵國(guó)和親昨寞。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,494評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容