一、 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地址