KVC 鍵值編碼(Key-value coding)
KVC是一套方便我們用字符串來操作對象的機(jī)制,可以使得操作對象時(shí)跟操作字典一樣的靈活。在字典轉(zhuǎn)模型的領(lǐng)域中應(yīng)用起來極為方便损同,并且KVC可以輕松的幫我們突破訪問限制的一些問題殴蓬,直接訪問到私有成員
- 主要方法
- (id)valueForKey:(NSString *)key;
- (void)setValue:(id)value forKey:(NSString *)key;
//如果需要操作訪問一些“屬性里的屬性”時(shí)侣背,就用帶Path的方法來操作
- (id)valueForKeyPath:(NSString *)keyPath;
- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
- KVC的優(yōu)點(diǎn)
(1). 無論類中的成員是否私有磁椒,用KVC都可以對它們正常取值和賦值
比如當(dāng)屬性在擴(kuò)展里面 無法通過點(diǎn)語法賦值 可以通過kvc賦值
(2). 不管你的成員變量是否加下劃線,你用KVC取值和賦值時(shí)傳入的屬性名都可以不帶下劃線
(3). 大大簡化字典轉(zhuǎn)模型代碼,KVC提供了一套更簡潔的操作方式,只需你傳入一個(gè)字典慎王,就可以幫你自動(dòng)把字典里的每一項(xiàng)賦值給你實(shí)體類對應(yīng)的屬性
NSDictionary *dic = @{
@"name":@"test",
//KVC中蚓土,賦值時(shí)傳入的值都只能是對象類型,無法直接傳入基本數(shù)據(jù)類型
@"age":@18,
@"address":@"China"
};
[p setValuesForKeysWithDictionary:dic];
NSLog(@"name:%@ age:%ld address:%@",p.name,p.age,p.address);
KVC缺點(diǎn):
1.在編碼時(shí)很容易輸錯(cuò)key導(dǎo)致問題
解決辦法:用KVC時(shí)傳入的Key必須保證類中存在同名的屬性柬祠。否則會(huì)運(yùn)行時(shí)崩潰北戏。如果不希望運(yùn)行時(shí)直接崩潰,就需要在類里重寫setValue:值 forUndefinedKey:鍵方法漫蛔,這樣,當(dāng)用KVC對Person對象賦值了一個(gè)Key與屬性對應(yīng)不上的錯(cuò)誤時(shí)旧蛾,系統(tǒng)會(huì)自動(dòng)調(diào)用這個(gè)方法.
2.語法相較點(diǎn)語法而言也略微繁瑣莽龟。
解決辦法:用runtime取代實(shí)現(xiàn)方法:
KVC運(yùn)用了一個(gè)isa-swizzling技術(shù),任何對象都有isa指針锨天。KVC主要通過isa-swizzling毯盈,來實(shí)現(xiàn)其內(nèi)部查找定位的:
(1) 實(shí)例方法調(diào)用時(shí),通過對象的 isa 在類中獲取方法的實(shí)現(xiàn)
(2) 類方法調(diào)用時(shí)病袄,通過類的 isa 在元類中獲取方法的實(shí)現(xiàn)KVC的賦值原理:
@implementation Person {
NSString *_address;
}
Person *p =[[Person alloc] init];
[p setValue:@"China" forKey:@"address"];
NSString *ads = [p valueForKey:@"address"];
找的順序是 setAddress:==>_address==>_isAddress==>address==>isAddress
- 用KVC賦值時(shí)搂赋,會(huì)優(yōu)先調(diào)用set<Key>:屬性值方法(setAddress:(NSString *)address)
- 如果找不到,則會(huì)先找?guī)聞澗€的成員變量_<key>(_address)益缠,再找_is<Key>(_isAddress)脑奠,如果找到則賦值
- 如果上面都找不到,則會(huì)先查找不帶下劃線的成員變量<key>(address)幅慌,再找is<Key>(isAddress)宋欺,找到則賦值
- 如果上面列出的方法或者成員變量都不存在,系統(tǒng)將會(huì)執(zhí)行該對象的setValue:forUNdefinedKey:方法胰伍,默認(rèn)是拋出異常
5.如果想讓這個(gè)類禁用KVC齿诞,那么重寫+ (BOOL)accessInstanceVariablesDirectly方法讓其返回NO即可,這樣的話如果KVC沒有找到set<Key>:屬性名時(shí)骂租,會(huì)直接用setValue:forUndefinedKey:方法.
KVO 鍵值觀察Key-value observing)
- 實(shí)現(xiàn)原理
當(dāng)你觀察一個(gè)對象時(shí)祷杈,一個(gè)新的類會(huì)動(dòng)態(tài)被創(chuàng)建。這個(gè)類繼承自該對象的原本的類渗饮,并重寫了被觀察屬性的 setter 方法但汞。自然,重寫的 setter 方法會(huì)負(fù)責(zé)在調(diào)用原 setter 方法之前和之后抽米,通知所有觀察對象值的更改特占。最后把這個(gè)對象的 isa 指針 ( isa 指針告訴 Runtime 系統(tǒng)這個(gè)對象的類是什么 ) 指向這個(gè)新創(chuàng)建的子類,對象就神奇的變成了新創(chuàng)建的子類的實(shí)例云茸。
原來是目,這個(gè)中間類,繼承自原本的那個(gè)類标捺。不僅如此懊纳,Apple 還重寫了 -class 方法揉抵,企圖欺騙我們這個(gè)類沒有變,就是原本那個(gè)類嗤疯。
簡單的說就是在給某個(gè)類的屬性添加觀察的時(shí)候冤今,系統(tǒng)就創(chuàng)建一個(gè)該類派生類,然后派生類重寫該屬性的setter方法實(shí)現(xiàn)觀察對象的改變茂缚,然后將這個(gè)對象的 isa 指針指向這個(gè)派生類戏罢。對象就變成了新創(chuàng)建的子類的實(shí)例。
它是基于KVC實(shí)現(xiàn)的脚囊,實(shí)現(xiàn)了響應(yīng)式編程龟糕,內(nèi)部實(shí)現(xiàn)運(yùn)用了runtime的特性。
下面這篇文章介紹了如何手動(dòng)實(shí)現(xiàn)KVO
http://tech.glowing.com/cn/implement-kvo/