由于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)聽者舔箭,這里需要注意罩缴。