iOS KVC與KVO簡介

QiShare發表於2019-03-01

級別: ★★☆☆☆
標籤:「iOS」「KVC」「KVO」
作者: dac_1033
審校: QiShare團隊

一、 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:使用時,addObserverremoveObserver需要成對出現,在銷燬物件時移除監聽。

工程原始碼:GitHub地址

推薦文章:
iOS 本地化(IB篇)
iOS 本地化(非IB篇)
iOS 檔案操作簡介
iOS 關鍵幀動畫
奇舞週刊

相關文章