OC動態(tài)特性之 — KVC丰刊、KVO

由于Objective-C是基于Smalltalk進(jìn)行設(shè)計(jì)的,所以它具有動態(tài)加載蝉稳、動態(tài)綁定等特性。Key-value coding (KVC) 和 key-value observing (KVO) 是兩種能讓我們駕馭 Objective-C 動態(tài)特性并簡化代碼的機(jī)制掘鄙。

1.KVC

在ObjC的編程中耘戚,我們習(xí)慣于通過屬性的set和get方法來對屬性的值進(jìn)行讀寫,其實(shí)由于ObjC的語言特性操漠,你根本不必進(jìn)行任何操作就可以進(jìn)行屬性的動態(tài)讀寫收津,這種方式就是Key Value Coding(簡稱KVC)。

KVC的操作方法由NSKeyValueCoding協(xié)議提供浊伙,而NSObject就實(shí)現(xiàn)了這個(gè)協(xié)議撞秋,也就是說ObjC中幾乎所有的對象都支持KVC操作,常用的KVC操作方法如下:

寫方法:setValue:屬性值 forKey:屬性名(用于簡單路徑)嚣鄙、setValue:屬性值 forKeyPath:屬性路徑(用于復(fù)合路徑吻贿,例如Person有一個(gè)Account類型的屬性,那么person.account就是一個(gè)復(fù)合屬性)

讀方法:valueForKey:屬性名(簡單路徑)哑子、valueForKeyPath:屬性名(復(fù)合路徑)

示例代碼:

Book.h

#import

@interfaceBook:NSObject{

// price屬性

double_price;

}

@end

Book.m

#import "Book.h"

@implementationBook

@end

Person.h

#import

// 聲明Book類

@classBook;

#pragma屬性:無set和get方法

@interfacePerson:NSObject{

int_age;

Book*_book;

}

#pragma屬性:有set和get方法

@property(nonatomic,copy)NSString*name;

@property(nonatomic,assign)floatheight;

@end

Person.m

#import "Person.h"

@implementationPerson

@end

main.m

#import

#import "Person.h"

#import "Book.h"

intmain(intargc,constchar*argv[]){

@autoreleasepool{

Person*person=[[Personalloc]init];

Book*book=[[Bookalloc]init];

// setValue:forKey: 方法設(shè)置簡單屬性的值舅列,value值必須是OC對象

[person setValue:@18forKey:@"_age"];

[person setValue:@1.7forKey:@"height"];

[person setValue:@"jack"forKey:@"name"];

[person setValue:book forKey:@"_book"];

// setValue:forKeyPath: 方法設(shè)置復(fù)合屬性的值

[person setValue:@25.8forKeyPath:@"book.price"];

// valueForKey: 方法獲取簡單屬性的值

intage=[[person valueForKey:@"age"]intValue];

floatheight=person.height;

NSString*name=[person valueForKey:@"_name"];

// valueForKeyPath: 方法獲取復(fù)合屬性的值

doublebookPrice=[[person valueForKeyPath:@"_book._price"]doubleValue];

NSLog(@"age = %d",age);

NSLog(@"height = %f",height);

NSLog(@"name = %@",name);

NSLog(@"bookPrice = %f",bookPrice);

}

return0;

}

小結(jié):

如果是動態(tài)設(shè)置屬性,以上文 age 屬性為例卧蜓,會優(yōu)先考慮調(diào)用 setAge: 方法帐要,如果沒有該方法則優(yōu)先考慮搜索成員變量 _age,如果仍然不存在則搜索成員變量 age弥奸,如果最后仍然沒搜索到則會調(diào)用這個(gè)類的setValue:forUndefinedKey:方法(注意搜索過程中不管這些方法榨惠、成員變量是私有的還是公共的都能正確設(shè)置),所以key值加不加下劃線都是可以的其爵。

如果是動態(tài)讀取屬性冒冬,則優(yōu)先考慮調(diào)用 age 方法(屬性age的getter方法),如果沒有搜索到則會優(yōu)先搜索成員變量 _age摩渺,如果仍然不存在則搜索成員變量 age简烤,如果最后仍然沒搜索到則會調(diào)用這個(gè)類的valueforUndefinedKey:方法(注意搜索過程中不管這些方法、成員變量是私有的還是公共的都能正確讀取)摇幻,所以key值加不加下劃線都是可以的横侦。

2.KVO

在如今比較流行的MVVM設(shè)計(jì)模式中,需要有一種雙向綁定的機(jī)制绰姻,在數(shù)據(jù)模型發(fā)生了修改之后立即將改變呈現(xiàn)到UI視圖上去枉侧。OC中原生的就支持這么一種機(jī)制,那就是Key Value Observing(簡稱KVO)狂芋。KVO其實(shí)是一種觀察者模式榨馁,利用它可以很容易實(shí)現(xiàn)視圖組件和數(shù)據(jù)模型的分離,當(dāng)數(shù)據(jù)模型的屬性值改變之后作為監(jiān)聽器的視圖組件就會被激發(fā)帜矾,激發(fā)時(shí)就會回調(diào)監(jiān)聽器自身翼虫。在ObjC中要實(shí)現(xiàn)KVO則必須實(shí)現(xiàn)NSKeyValueObServing協(xié)議屑柔,不過幸運(yùn)的是NSObject已經(jīng)實(shí)現(xiàn)了該協(xié)議,因此幾乎所有的ObjC對象都可以使用KVO珍剑。

在ObjC中使用KVO操作常用的方法如下:

注冊指定Key路徑的監(jiān)聽器:addObserver: forKeyPath: options:? context:

刪除指定Key路徑的監(jiān)聽器:removeObserver: forKeyPath掸宛、removeObserver: forKeyPath: context:

回調(diào)監(jiān)聽:observeValueForKeyPath: ofObject: change: context:

KVO的使用步驟也比較簡單:

通過addObserver: forKeyPath: options: context:為被監(jiān)聽對象(它通常是數(shù)據(jù)模型)注冊監(jiān)聽器

重寫監(jiān)聽器的observeValueForKeyPath: ofObject: change: context:方法

這里我們還是在上面的例子基礎(chǔ)上繼續(xù)擴(kuò)展,我們?yōu)?person 對象添加一個(gè)監(jiān)聽者 observer招拙。當(dāng)我們的 person 對象的 height 屬性值變動之后我們希望 observer 可以及時(shí)獲得通知唧瘾。

為了認(rèn)識KVO能監(jiān)聽對象屬性值的哪幾種方式的變化,我們自己實(shí)現(xiàn)一個(gè)方法來改變本身屬性的值别凤,那么在Person類中做一些改變饰序,添加一個(gè)changValue方法:

-(void)changeValue{

_height+=0.1;

NSLog(@"----- 自己實(shí)現(xiàn)的方法改變height的值 ------");

NSLog(@"height = %f",_height);

}

創(chuàng)建一個(gè)Observer類:

Observer.h

#import

@interfaceObserver:NSObject

@property(nonatomic,copy)NSString*name;

@end

Observer.m

#import "Observer.h"

@implementationObserver

// 這個(gè)方法在對象的監(jiān)聽屬性發(fā)生改變時(shí),會自動調(diào)用闻妓。監(jiān)聽者對屬性發(fā)生的改變做出什么反應(yīng)也體現(xiàn)在這里菌羽。

-(void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)objectchange:(NSDictionary*)change context:(void*)context{

NSLog(@"------ %@ 在監(jiān)聽 ------",self.name);

NSLog(@"keyPath: %@",keyPath);

NSLog(@"object: %@",[objectvalueForKey:@"name"]);

NSLog(@"change: %@",change);

NSLog(@"context: %@",context);

}

@end

main.m

#import

#import "Person.h"

#import "Observer.h"

intmain(intargc,constchar*argv[]){

@autoreleasepool{

// 創(chuàng)建一個(gè)person對象,設(shè)置兩個(gè)屬性值

Person*person=[[Personalloc]init];

person.height=1.7;

person.name=@"jack";

// 創(chuàng)建一個(gè)observer對象由缆,設(shè)置一個(gè)name屬性的值

Observer*observer=[[Observeralloc]init];

observer.name=@"observer";

// 為person對象注冊一個(gè)監(jiān)聽者

/**

*? 第1個(gè)參數(shù):誰來監(jiān)聽

*? 第2個(gè)參數(shù):監(jiān)聽哪一個(gè)屬性

*? 第3個(gè)參數(shù):屬性發(fā)生了什么變化

*? 第4個(gè)參數(shù):額外傳入的參數(shù)

*/

[person addObserver:observer forKeyPath:@"height"options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOldcontext:@"這里可以傳入一些東西..."];

// 通過setter方法改變被監(jiān)聽的屬性的值

person.height=1.8;

// 通過KVC方法改變被監(jiān)聽的屬性的值

[person setValue:@1.9forKey:@"height"];

// 通過自己實(shí)現(xiàn)的changeValue方法改變被監(jiān)聽的屬性的值

[person changeValue];

// 移除監(jiān)聽(在新版本編譯器中注祖,必須配對調(diào)用監(jiān)聽和移除監(jiān)聽的方法,否則程序會崩潰)

/**

*? 第1個(gè)參數(shù): 要移除哪個(gè)監(jiān)聽者

*? 第2個(gè)參數(shù): 監(jiān)聽的是哪個(gè)屬性

*/

[person removeObserver:observer forKeyPath:@"height"];

}

return0;

}

從上面的運(yùn)行結(jié)果來看均唉,只有通過setter或KVC修改的屬性值是晨,才會調(diào)用observeValueForKeyPath:方法,通過其他方式修改屬性值并不能通知監(jiān)聽者舔箭,這里需要注意罩缴。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市层扶,隨后出現(xiàn)的幾起案子箫章,更是在濱河造成了極大的恐慌,老刑警劉巖镜会,帶你破解...
    沈念sama閱讀 216,372評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件檬寂,死亡現(xiàn)場離奇詭異,居然都是意外死亡戳表,警方通過查閱死者的電腦和手機(jī)桶至,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來匾旭,“玉大人镣屹,你說我怎么就攤上這事〖劾裕” “怎么了女蜈?”我有些...
    開封第一講書人閱讀 162,415評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我伪窖,道長吏廉,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,157評論 1 292
  • 正文 為了忘掉前任惰许,我火速辦了婚禮,結(jié)果婚禮上史辙,老公的妹妹穿的比我還像新娘汹买。我一直安慰自己,他們只是感情好聊倔,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,171評論 6 388
  • 文/花漫 我一把揭開白布晦毙。 她就那樣靜靜地躺著,像睡著了一般耙蔑。 火紅的嫁衣襯著肌膚如雪见妒。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,125評論 1 297
  • 那天甸陌,我揣著相機(jī)與錄音须揣,去河邊找鬼。 笑死钱豁,一個(gè)胖子當(dāng)著我的面吹牛耻卡,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播牲尺,決...
    沈念sama閱讀 40,028評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼卵酪,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了谤碳?” 一聲冷哼從身側(cè)響起溃卡,我...
    開封第一講書人閱讀 38,887評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎蜒简,沒想到半個(gè)月后瘸羡,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,310評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡臭蚁,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,533評論 2 332
  • 正文 我和宋清朗相戀三年最铁,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片垮兑。...
    茶點(diǎn)故事閱讀 39,690評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡冷尉,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出系枪,到底是詐尸還是另有隱情雀哨,我是刑警寧澤,帶...
    沈念sama閱讀 35,411評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站雾棺,受9級特大地震影響膊夹,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜捌浩,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,004評論 3 325
  • 文/蒙蒙 一放刨、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧尸饺,春花似錦进统、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至迹栓,卻和暖如春掉分,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背克伊。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評論 1 268
  • 我被黑心中介騙來泰國打工酥郭, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人答毫。 一個(gè)月前我還...
    沈念sama閱讀 47,693評論 2 368
  • 正文 我出身青樓褥民,卻偏偏與公主長得像,于是被迫代替她去往敵國和親洗搂。 傳聞我的和親對象是個(gè)殘疾皇子消返,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,577評論 2 353

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

  • 20- 枚舉,枚舉原始值,枚舉相關(guān)值,switch提取枚舉關(guān)聯(lián)值 Swift枚舉: Swift中的枚舉比OC中的枚...
    iOS_恒仔閱讀 2,278評論 1 6
  • KCV 其實(shí)由于ObjC的語言特性,你根部不必進(jìn)行任何操作就可以進(jìn)行屬性的動態(tài)讀寫耘拇,這種方式就是Key Value...
    TYM閱讀 1,059評論 0 4
  • 目錄:1.KVC用法撵颊;2.KVC和對象的setter、getter方法的區(qū)別惫叛;3.key和keyPath的區(qū)別倡勇;4...
    倫倫子_f7b3閱讀 574評論 0 1
  • Swift 介紹 簡介 Swift 語言由蘋果公司在 2014 年推出,用來撰寫 OS X 和 iOS 應(yīng)用程序 ...
    大L君閱讀 3,209評論 3 25
  • 如果一個(gè)人正在用心去做一件事情嘉涌,但是他不知道結(jié)果會是怎么樣妻熊,其實(shí)他的內(nèi)心是非常糾結(jié),同時(shí)也是非常期待的仑最。當(dāng)然啦扔役,如...
    未自己閱讀 142評論 0 0