一、 KVC
1.1 KVC介紹
KVC是Key Value Coding
的縮寫,即鍵值編碼。
在iOS的開發中,可以通過key
名直接訪問實例對象的屬性,而不需要調用明確的存取方法。
也就是說我們可以在程序運行時動態地訪問和修改對象的屬性。
KVC的定義都是對NSObject
的擴展來實現的NSObject(NSKeyValueCoding)
,所以對于所有繼承了NSObject
的類型,都能使用KVC
。(PS:一些純Swift
類和結構體是不支持KVC
的,因為沒有繼承NSObject
)
1.2 KVC中的常用方法
在程序開發過程中,最常用的KVC方法有四個:
// 通過Key來獲取對象屬性的值
- (nullable id)valueForKey:(NSString *)key;
// 通過Key來設置對象屬性的值
- (void)setValue:(nullable id)value forKey:(NSString *)key;
// 通過復合路徑KeyPath來獲取對象屬性的值
- (nullable id)valueForKeyPath:(NSString *)keyPath;
// 通過復合路徑KeyPath來設置對象屬性的值
- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath;
1.3 KVC的使用
首先,我們定義兩個簡單的類,在代碼中簡單使用一下上面的方法:
- 員工類QiStaff.h:
@interface QiStaff : NSObject
// 員工編號
@property (nonatomic, strong) NSString *staffId;
// 員工名字
@property (nonatomic, strong) NSString *staffName;
@end
- 公司類QiCompany.h
#import <Foundation/Foundation.h>
#import "QiStaff.h"
@interface QiCompany : NSObject
// 公司名字
@property (nonatomic, strong) NSString *name;
// 員工變量
@property (nonatomic, strong) QiStaff *staff;
@end
- 公司類QiCompany.m:
#import "QiCompany.h"
@interface QiCompany()
@property (nonatomic, strong) NSString *addr;
@end
@implementation QiCompany
@end
- KVC方法的具體使用過程:
- (void)aboutKVC {
QiCompany *company=[[QiCompany alloc]init];
[company setValue:@"QiShare" forKey:@"name"];
[company setValue:@"北京市朝陽區酒仙橋路" forKey:@"addr"];
NSLog(@"公司:%@ 地址:%@", company.name, [company valueForKey:@"addr"]);
QiStaff *staff = [[QiStaff alloc]init];
company.staff = staff;
[company setValue:@"1000119" forKeyPath:@"staff.staffId"];
[company setValue:@"佩奇" forKeyPath:@"staff.staffName"];
NSLog(@"員工id:%@ 名字:%@", [company valueForKeyPath:@"staff.staffId"], [company valueForKeyPath:@"staff.staffName"]);
}
日志輸出:
2018-11-13 13:55:32.797847+0800 QiKVO&KVC[19722:585309] 公司:QiShare 地址:北京市朝陽區酒仙橋路
2018-11-13 13:55:32.798040+0800 QiKVO&KVC[19722:585309] 員工id:1000119 名字:佩奇
注:
1)方法中的KeyPath
用于“復合路徑”,例如QiCompany
類中有一個QiStaff
類型的屬性,則company.staff
就是一個“復合路徑”;
2)KVC
可以操作對象的私有變量。
二、KVO
2.1 KVO介紹
KVO 是Key Value Observing
的縮寫,即鍵值監聽。
鍵值監聽是典型的基于觀察者模式的應用。顧名思義,KVO可以通過監聽key
,來獲得value
的變化,用來在對象之間監聽狀態變化。
KVO的定義都是對NSObject
的擴展來實現的NSObject(NSKeyValueObserving)
,所以對于所有繼承了NSObject
的類型,都能使用KVO
。
2.2 KVO中的常用方法
在開發過程中常用的KVO方法有兩個,分別是添加觀察者和移除觀察者方法:
// 添加觀察者
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(void *)context;
// 移除觀察者
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;
// 代理 監聽值變的方法
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context;
NSKeyValueObservingOptions | 含義 |
---|---|
NSKeyValueObservingOptionNew | change字典中包括改變后的值。 |
NSKeyValueObservingOptionOld | change字典中包括改變前的值。 |
NSKeyValueObservingOptionInitial | 每次修改變量值時均觸發KVO通知。(包括添加觀察者之前的修改動作,而change字典中無相應的key-value) |
NSKeyValueObservingOptionPrior | 修改變量值之前和之后均立即觸發KVO通知。(兩次觸發中,chang字典的內容不同,修改之前的通知change字典中有notificationIsPrior = 1鍵值對) |
注:在我們日常開發中,我們一般取這個Options值為
NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld
,這時,當變量值被修改時,我們在change字典中既能得到新值,也能得到修改之前的值。
2.3 KVO的使用
我們在第一節所用的代碼基礎上進行修改,示例代碼如下:
- 員工類QiStaff.h:
#import <Foundation/Foundation.h>
#import "QiStaff.h"
@interface QiCompany : NSObject
// 公司名字
@property (nonatomic, strong) NSString *name;
// 員工變量
@property (nonatomic, strong) QiStaff *staff;
@end
- 公司類QiCompany.h:
#import <Foundation/Foundation.h>
#import "QiStaff.h"
@interface QiCompany : NSObject
// 公司名字
@property (nonatomic, strong) NSString *name;
// 員工變量
@property (nonatomic, strong) QiStaff *staff;
- (void)aboutKVO;
@end
- 公司類QiCompany.m:
#import "QiCompany.h"
@interface QiCompany()
@property (nonatomic, strong) NSString *addr;
@end
@implementation QiCompany
- (void)aboutKVO {
self.staff = [[QiStaff alloc] init];
self.staff.staffId = @"1000119";
self.staff.staffName = @"佩奇";
[self.staff addObserver:self forKeyPath:@"staffId" options:NSKeyValueObservingOptionNew context:nil];
[self.staff addObserver:self forKeyPath:@"staffName" options:NSKeyValueObservingOptionNew context:nil];
self.staff.staffId = @"1000120";
self.staff.staffName = @"佩德羅";
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
NSLog(@"keyPath = %@ object=%@ newValue=%@ context=%@", keyPath, object, [change objectForKey:@"new"], context);
}
- (void)dealloc {
[self.staff removeObserver:self forKeyPath:@"staffId"];
[self.staff removeObserver:self forKeyPath:@"staffName"];
}
@end
日志輸出:
2018-11-13 16:25:21.215872+0800 QiKVO&KVC[20986:895803] keyPath = staffId object=<QiStaff: 0x6000017286c0> newValue=1000120 context=(null)
2018-11-13 16:25:21.216040+0800 QiKVO&KVC[20986:895803] keyPath = staffName object=<QiStaff: 0x6000017286c0> newValue=佩德羅 context=(null)
PS:使用時,
addObserver
與removeObserver
需要成對出現,在銷毀對象時移除監聽。
工程源碼:GitHub地址