1谜洽、KVC賦值
// 1.1 創(chuàng)建人
PTLPerson *p = [[PTLPerson alloc] init];
self.person = p;
// 1.2 創(chuàng)建狗
PTLDog *dog = [[PTLDog alloc] init];
// 1.3 將狗賦值給人
[p setValue:dog forKeyPath:@"dog"];
// 1.4 通過KVC給dog的weight屬性賦值
賦值時(shí)會(huì)自動(dòng)找到人擁有的dog的weight屬性
[p setValue:@10.0 forKeyPath:@"dog.weight"];
NSLog(@"books = %@", [p valueForKeyPath:@"dog.weight"]);
[dog print];
2萝映、 KVC取值
NSMutableArray *tempM = [NSMutableArray array];
// 2.1 kvc取出出數(shù)組books中price的值
for (PTLBook *book in [p valueForKeyPath:@"books"]) {
[tempM addObject:[book valueForKeyPath:@"price"]];
}
NSLog(@"%@", tempM);
// 2.2 kvc取出數(shù)組中price的最大值
NSLog(@"Max = %@", [[p valueForKeyPath:@"books"] valueForKeyPath:@"@max.price"]);
3、 KVO原理
- KVO 是 Objective-C 對(duì)觀察者設(shè)計(jì)模式的一種實(shí)現(xiàn),另外一種是:通知機(jī)制(notification)
KVO提供一種機(jī)制,指定一個(gè)被觀察對(duì)象(例如A類),當(dāng)對(duì)象某個(gè)屬性(例如A中的字符串name)發(fā)生更改時(shí),對(duì)象會(huì)獲得通知,并作出相應(yīng)處理
在MVC設(shè)計(jì)架構(gòu)下的項(xiàng)目,KVO機(jī)制很適合實(shí)現(xiàn)mode模型和controller之間的通訊阐虚。
例如:代碼中,在模型類A創(chuàng)建屬性數(shù)據(jù),在控制器中創(chuàng)建觀察者,一旦屬性數(shù)據(jù)發(fā)生改變就收到觀察者收到通知,通過KVO再在控制器使用回調(diào)方法處理實(shí)現(xiàn)視圖B的更新;(本文中的應(yīng)用就是這樣的例子.)
KVO在Apple中的API文檔如下:
Automatic key-value observing is implemented using a technique called isa-swizzling… When an observer is registered for an attribute of an object the isa pointer of the observed object is modified, pointing to an intermediate class rather than at the true class …
KVO 的實(shí)現(xiàn)依賴于 Objective-C 強(qiáng)大的 Runtime【 ,從以上Apple 的文檔可以看出蘋果對(duì)于KVO機(jī)制的實(shí)現(xiàn)是一筆帶過,而具體的細(xì)節(jié)沒有過多的描述,但是我們可以通過Runtime的所提供的方法去探索關(guān)于KVO機(jī)制的底層實(shí)現(xiàn)原理.當(dāng)觀察某對(duì)象 A 時(shí)序臂,KVO 機(jī)制動(dòng)態(tài)創(chuàng)建一個(gè)對(duì)象A當(dāng)前類的子類,并為這個(gè)新的子類重寫了被觀察屬性 keyPath 的 setter 方法实束。setter 方法隨后負(fù)責(zé)通知觀察對(duì)象屬性的改變狀況奥秆。
Apple 使用了 isa 混寫(isa-swizzling)來實(shí)現(xiàn) KVO 。當(dāng)觀察對(duì)象A時(shí)咸灿,KVO機(jī)制動(dòng)態(tài)創(chuàng)建一個(gè)新的名為:NSKVONotifying_A 的新類构订,該類繼承自對(duì)象A的本類,且 KVO 為 NSKVONotifying_A 重寫觀察屬性的 setter 方法避矢,setter 方法會(huì)負(fù)責(zé)在調(diào)用原 setter 方法之前和之后悼瘾,通知所有觀察對(duì)象屬性值的更改情況。
NSKVONotifying_A類剖析:在這個(gè)過程,被觀察對(duì)象的 isa 指針從指向原來的A類,被KVO機(jī)制修改為指向系統(tǒng)新創(chuàng)建的子類 NSKVONotifying_A類,來實(shí)現(xiàn)當(dāng)前類屬性值改變的監(jiān)聽;
所以當(dāng)我們從應(yīng)用層面上看來,完全沒有意識(shí)到有新的類出現(xiàn),這是系統(tǒng)“隱瞞”了對(duì)KVO的底層實(shí)現(xiàn)過程,讓我們誤以為還是原來的類审胸。但是此時(shí)如果我們創(chuàng)建一個(gè)新的名為“NSKVONotifying_A”的類(),就會(huì)發(fā)現(xiàn)系統(tǒng)運(yùn)行到注冊KVO的那段代碼時(shí)程序就崩潰,因?yàn)橄到y(tǒng)在注冊監(jiān)聽的時(shí)候動(dòng)態(tài)創(chuàng)建了名為NSKVONotifying_A的中間類,并指向這個(gè)中間類了亥宿。
因而在該對(duì)象上對(duì) setter 的調(diào)用就會(huì)調(diào)用已重寫的 setter,從而激活鍵值通知機(jī)制。
- 子類setter方法剖析:KVO的鍵值觀察通知依賴于 NSObject 的兩個(gè)方法:willChangeValueForKey:和 didChangevlueForKey:,在存取數(shù)值的前后分別調(diào)用2個(gè)方法:
被觀察屬性發(fā)生改變之前,willChangeValueForKey:被調(diào)用,通知系統(tǒng)該 keyPath 的屬性值即將變更;當(dāng)改變發(fā)生后, didChangeValueForKey: 被調(diào)用,通知系統(tǒng)該 keyPath 的屬性值已經(jīng)變更;
之后observeValueForKey:ofObject:change:context: 也會(huì)被調(diào)用砂沛。且重寫觀察屬性的setter 方法這種繼承方式的注入是在運(yùn)行時(shí)而不是編譯時(shí)實(shí)現(xiàn)的烫扼。
KVO為子類的觀察者屬性重寫調(diào)用存取方法的工作原理在代碼中相當(dāng)于:
-(void)setName:(NSString *)newName
{
[self willChangeValueForKey:@"name"]; //KVO在調(diào)用存取方法之前總調(diào)用
[super setValue:newName forKey:@"name"]; //調(diào)用父類的存取方法
[self didChangeValueForKey:@"name"]; //KVO在調(diào)用存取方法之后總調(diào)用
}
更多:iOS面試題合集