iOS 中KVC痹扇、KVO帘营、NSNotification逐哈、delegate 總結(jié)及區(qū)別

c1昂秃、KVC,即是指 NSKeyValueCoding算途,一個(gè)非正式的Protocol嘴瓤,提供一種機(jī)制來(lái)間接訪問(wèn)對(duì)象的屬性。而不是通過(guò)調(diào)用Setter筛谚、Getter方法訪問(wèn)停忿。KVO 就是基于 KVC 實(shí)現(xiàn)的關(guān)鍵技術(shù)之一席赂。

Demo:

@interface myPerson : NSObject

{

NSString*_name;

int? ? ? _age;

int? ? ? _height;

int? ? ? _weight;

} @end

@interface testViewController :UIViewController

@property (nonatomic, retain) myPerson*testPerson;

@end

- (void)testKVC

{

testPerson = [[myPerson alloc] init];

NSLog(@"testPerson‘s init height =%@", [testPerson valueForKey:@"height"]);

[testPerson

setValue:[NSNumber

numberWithInt:168]forKey:@"height"];? ? NSLog(@"testPerson‘s height =

%@", [testPerson valueForKey:@"height"]);

}

第一段代碼是定義了一個(gè)myPerson的類颅停,這個(gè)類有一個(gè)_height的屬性便监,但是沒(méi)有提供任何getter/setter的訪問(wèn)方法。同時(shí)在testViewController這個(gè)類里面有一個(gè)myPerson的對(duì)象指針毁靶。

當(dāng)myPerson實(shí)例化后预吆,常規(guī)來(lái)說(shuō)是無(wú)法訪問(wèn)這個(gè)對(duì)象的_height屬性的胳泉,不過(guò)通過(guò)KVC我們做到了扇商,代碼就是testKVC這個(gè)函數(shù)。

運(yùn)行之后打印值就是:

2015-3-13 11:16:21.970 test[408:c07] testPerson‘s init height = 0

2015-3-13 11:16:21.971 test[408:c07] testPerson‘s height = 168

這就說(shuō)明確實(shí)讀寫了_height屬性蔬芥。

KVC的常用方法:

- (id)valueForKey:(NSString *)key; -(void)setValue:(id)value forKey:(NSString *)key;

valueForKey的方法根據(jù)key的值讀取對(duì)象的屬性笔诵,setValue:forKey:是根據(jù)key的值來(lái)寫對(duì)象的屬性乎婿。

注意:

(1). key的值必須正確街佑,如果拼寫錯(cuò)誤,會(huì)出現(xiàn)異常

(2). 當(dāng)key的值是沒(méi)有定義的剂公,valueForUndefinedKey:這個(gè)方法會(huì)被調(diào)用吊宋,如果你自己寫了這個(gè)方法璃搜,key的值出錯(cuò)就會(huì)調(diào)用到這里來(lái)

(3). 因?yàn)轭恔ey反復(fù)嵌套这吻,所以有個(gè)keyPath的概念篙议,keyPath就是用.號(hào)來(lái)把一個(gè)一個(gè)key鏈接起來(lái),這樣就可以根據(jù)這個(gè)路徑訪問(wèn)下去

(4). NSArray/NSSet等都支持KVC

2移怯、KVO的是KeyValue Observe的縮寫舟误,中文是鍵值觀察姻乓。這是一個(gè)典型的觀察者模式蹋岩,觀察者在鍵值改變時(shí)會(huì)得到通知剪个。iOS中有個(gè)Notification的機(jī)制禁偎,也可以獲得通知,但這個(gè)機(jī)制需要有個(gè)Center笆檀,相比之下KVO更加簡(jiǎn)潔而直接。

KVO的使用也很簡(jiǎn)單盒至,就是簡(jiǎn)單的3步酗洒。

1.注冊(cè)需要觀察的對(duì)象的屬性addObserver:forKeyPath:options:context:

2.實(shí)現(xiàn)observeValueForKeyPath:ofObject:change:context:方法士修,這個(gè)方法當(dāng)觀察的屬性變化時(shí)會(huì)自動(dòng)調(diào)用

3.取消注冊(cè)觀察removeObserver:forKeyPath:context:

Demo:

@interface myPerson : NSObject

{

NSString *_name;

int? ? ? _age;

int? ? ? _height;

int? ? ? _weight;

}

@end

@interface testViewController : UIViewController

@property (nonatomic, retain) myPerson *testPerson;

- (IBAction)onBtnTest:(id)sender;

@end

- (void)testKVO

{

testPerson = [[myPerson alloc] init];

[testPerson addObserver:self forKeyPath:@"height" options:NSKeyValueObservingOptionNew context:nil];

}

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

{

if ([keyPath isEqualToString:@"height"]) {

NSLog(@"Height is changed! new=%@", [change valueForKey:NSKeyValueChangeNewKey]);

} else {

[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];

}

}

- (IBAction)onBtnTest:(id)sender {

int h = [[testPerson valueForKey:@"height"] intValue];

[testPerson setValue:[NSNumber numberWithInt:h+1] forKey:@"height"];

NSLog(@"person height=%@", [testPerson valueForKey:@"height"]);

}

- (void)dealloc

{

[testPerson removeObserver:self forKeyPath:@"height" context:nil];

[super dealloc];

}

第一段代碼聲明了myPerson類,里面有個(gè)_height的屬性樱衷。在testViewController有一個(gè)testPerson的對(duì)象指針棋嘲。

在testKVO這個(gè)方法里面,我們注冊(cè)了testPerson這個(gè)對(duì)象height屬性的觀察矩桂,這樣當(dāng)testPerson的height屬性變化時(shí)沸移, 會(huì)得到通知侄榴。在這個(gè)方法中還通過(guò)NSKeyValueObservingOptionNew這個(gè)參數(shù)要求把新值在dictionary中傳遞過(guò)來(lái)雹锣。

重寫了observeValueForKeyPath:ofObject:change:context:方法,這個(gè)方法里的change這個(gè)NSDictionary對(duì)象包含了相應(yīng)的值癞蚕。

需要強(qiáng)調(diào)的是KVO的回調(diào)要被調(diào)用蕊爵,屬性必須是通過(guò)KVC的方法來(lái)修改的,如果是調(diào)用類的其他方法來(lái)修改屬性桦山,這個(gè)觀察者是不會(huì)得到通知的攒射。

3、NSNotification的用法見(jiàn)http://blog.csdn.net/eduora_meimei/article/details/44198909

區(qū)別:

delegate優(yōu)勢(shì)

1.非常嚴(yán)格的語(yǔ)法恒水。所有將聽(tīng)到的事件必須是在delegate協(xié)議中有清晰的定義会放。

2.如果delegate中的一個(gè)方法沒(méi)有實(shí)現(xiàn)那么就會(huì)出現(xiàn)編譯警告/錯(cuò)誤

3.協(xié)議必須在controller的作用域范圍內(nèi)定義

4.在一個(gè)應(yīng)用中的控制流程是可跟蹤的并且是可識(shí)別的;

5.在一個(gè)控制器中可以定義定義多個(gè)不同的協(xié)議寇窑,每個(gè)協(xié)議有不同的delegates

6.沒(méi)有第三方對(duì)象要求保持/監(jiān)視通信過(guò)程鸦概。

7.能夠接收調(diào)用的協(xié)議方法的返回值。這意味著delegate能夠提供反饋信息給controller

缺點(diǎn)

1.需要定義很多代碼:1.協(xié)議定義甩骏;2.controller的delegate屬性窗市;3.在delegate本身中實(shí)現(xiàn)delegate方法定義

2.在釋放代理對(duì)象時(shí),需要小心的將delegate改為nil饮笛。一旦設(shè)定失敗咨察,那么調(diào)用釋放對(duì)象的方法將會(huì)出現(xiàn)內(nèi)存crash

3.在一個(gè)controller中有多個(gè)delegate對(duì)象,并且delegate是遵守同一個(gè)協(xié)議福青,但還是很難告訴多個(gè)對(duì)象同一個(gè)事件摄狱,不過(guò)有可能。

notification優(yōu)勢(shì)

1.不需要編寫多少代碼无午,實(shí)現(xiàn)比較簡(jiǎn)單媒役;

2.對(duì)于一個(gè)發(fā)出的通知,多個(gè)對(duì)象能夠做出反應(yīng)宪迟,即1對(duì)多的方式實(shí)現(xiàn)簡(jiǎn)單

3.controller能夠傳遞context對(duì)象(dictionary)酣衷,context對(duì)象攜帶了關(guān)于發(fā)送通知的自定義的信息

缺點(diǎn)

1.在編譯期不會(huì)檢查通知是否能夠被觀察者正確的處理;

2.在釋放注冊(cè)的對(duì)象時(shí)次泽,需要在通知中心取消注冊(cè)穿仪;

3.在調(diào)試的時(shí)候應(yīng)用的工作以及控制過(guò)程難跟蹤席爽;

4.需要第三方對(duì)喜愛(ài)那個(gè)來(lái)管理controller與觀察者對(duì)象之間的聯(lián)系;

5.controller和觀察者需要提前知道通知名稱啊片、UserInfodictionary keys只锻。如果這些沒(méi)有在工作區(qū)間定義,那么會(huì)出現(xiàn)不同步的情況紫谷;

6.通知發(fā)出后齐饮,controller不能從觀察者獲得任何的反饋信息。

KVO優(yōu)勢(shì)

1.能夠提供一種簡(jiǎn)單的方法實(shí)現(xiàn)兩個(gè)對(duì)象間的同步碴里。例如:model和view之間同步沈矿;

2.能夠?qū)Ψ俏覀儎?chuàng)建的對(duì)象,即內(nèi)部對(duì)象的狀態(tài)改變作出響應(yīng)咬腋,而且不需要改變內(nèi)部對(duì)象(SKD對(duì)象)的實(shí)現(xiàn);

3.能夠提供觀察的屬性的最新值以及先前值睡互;

4.用key paths來(lái)觀察屬性根竿,因此也可以觀察嵌套對(duì)象;

5.完成了對(duì)觀察對(duì)象的抽象就珠,因?yàn)椴恍枰~外的代碼來(lái)允許觀察值能夠被觀察

缺點(diǎn)

1.我們觀察的屬性必須使用strings來(lái)定義寇壳。因此在編譯器不會(huì)出現(xiàn)警告以及檢查;

2.對(duì)屬性重構(gòu)將導(dǎo)致我們的觀察代碼不再可用妻怎;

3.復(fù)雜的“IF”語(yǔ)句要求對(duì)象正在觀察多個(gè)值壳炎。這是因?yàn)樗械挠^察代碼通過(guò)一個(gè)方法來(lái)指向;

4.當(dāng)釋放觀察者時(shí)不需要移除觀察者逼侦。

1.? 效率肯定是delegate比NSNotification高匿辩。

delegate方法比notification更加直接,最典型的特征

是榛丢,delegate方法往往需要關(guān)注返回值铲球,也就是delegate方法的結(jié)果。比如-windowShouldClose:晰赞,需要關(guān)心返回的是yes

還是no稼病。所以delegate方法往往包含

should這個(gè)很傳神的詞。也就是好比你做我的delegate掖鱼,我會(huì)問(wèn)你我想關(guān)閉窗口你愿意嗎然走?你需要給我一個(gè)答案,我根據(jù)你的答案來(lái)決定如何做下一

步戏挡。相反的芍瑞,notification最大的特色就是不關(guān)心接受者的態(tài)度,我只管把通告放出來(lái)增拥,你接受不接受就是你的事情啄巧,同時(shí)我也不關(guān)心結(jié)果寻歧。所以

notification往往用did這個(gè)詞匯,比如NSWindowDidResizeNotification秩仆,那么NSWindow對(duì)象放出這個(gè)

notification后就什么都不管了也不會(huì)等待接

受者的反應(yīng)码泛。

2、KVO和NSNotification的區(qū)別:

和delegate一樣澄耍,KVO和NSNotification的作用也是類與類之間的通信噪珊,與delegate不同的是1)這兩個(gè)都是負(fù)責(zé)發(fā)出通知,剩下的事情就不管了齐莲,所以沒(méi)有返回值痢站;2)delegate只是一對(duì)一,而這兩個(gè)可以一對(duì)多选酗。這兩者也有各自的特點(diǎn)阵难。

iOS 中KVC、KVO芒填、NSNotification呜叫、delegate 總結(jié)及區(qū)別

標(biāo)簽:iphone開(kāi)發(fā)kvc? kvo? delegate

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市殿衰,隨后出現(xiàn)的幾起案子朱庆,更是在濱河造成了極大的恐慌,老刑警劉巖闷祥,帶你破解...
    沈念sama閱讀 218,451評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件娱颊,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡凯砍,警方通過(guò)查閱死者的電腦和手機(jī)箱硕,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)果覆,“玉大人颅痊,你說(shuō)我怎么就攤上這事【执” “怎么了斑响?”我有些...
    開(kāi)封第一講書人閱讀 164,782評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)钳榨。 經(jīng)常有香客問(wèn)我舰罚,道長(zhǎng),這世上最難降的妖魔是什么薛耻? 我笑而不...
    開(kāi)封第一講書人閱讀 58,709評(píng)論 1 294
  • 正文 為了忘掉前任营罢,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘饲漾。我一直安慰自己蝙搔,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,733評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布考传。 她就那樣靜靜地躺著吃型,像睡著了一般。 火紅的嫁衣襯著肌膚如雪僚楞。 梳的紋絲不亂的頭發(fā)上勤晚,一...
    開(kāi)封第一講書人閱讀 51,578評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音泉褐,去河邊找鬼赐写。 笑死,一個(gè)胖子當(dāng)著我的面吹牛膜赃,可吹牛的內(nèi)容都是我干的挺邀。 我是一名探鬼主播,決...
    沈念sama閱讀 40,320評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼财剖,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼悠夯!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起躺坟,我...
    開(kāi)封第一講書人閱讀 39,241評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎乳蓄,沒(méi)想到半個(gè)月后咪橙,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,686評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡虚倒,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,878評(píng)論 3 336
  • 正文 我和宋清朗相戀三年美侦,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片魂奥。...
    茶點(diǎn)故事閱讀 39,992評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡菠剩,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出耻煤,到底是詐尸還是另有隱情具壮,我是刑警寧澤,帶...
    沈念sama閱讀 35,715評(píng)論 5 346
  • 正文 年R本政府宣布哈蝇,位于F島的核電站棺妓,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏炮赦。R本人自食惡果不足惜怜跑,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,336評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望吠勘。 院中可真熱鬧性芬,春花似錦峡眶、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,912評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至汽煮,卻和暖如春搏熄,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背暇赤。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,040評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工心例, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人鞋囊。 一個(gè)月前我還...
    沈念sama閱讀 48,173評(píng)論 3 370
  • 正文 我出身青樓止后,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親溜腐。 傳聞我的和親對(duì)象是個(gè)殘疾皇子译株,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,947評(píng)論 2 355

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