【iOS面試糧食】OC語言—KVC、KVO
本文章將記錄有關 KVC、KVO的特性,如有錯誤歡迎指出~
KVC(Key-Value Coding)鍵值編碼
基於Object-C的語言特性,KVC可以讓我們在開發中直接通過物件的字串引數(Key)獲取、賦值物件的屬性。那我們就可以通過KVC的特性來修改控制元件的私有屬性,是不是很刺激~
KVC的操作方法由NSKeyValueCoding
協議提供,而NSObject
就實現了這個協議,也就是說Object-C中幾乎所有的物件都支援KVC操作,常用的KVC操作方法如下:
- (nullable id)valueForKey:(NSString *)key; //直接通過屬性名來取值
- (void)setValue:(nullable id)value forKey:(NSString *)key; //通過屬性名來設值
- (nullable id)valueForKeyPath:(NSString *)keyPath; //通過屬性路徑來取值
- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath; //通過屬性路徑來設值
複製程式碼
接下來,我們看下通過屬性名設定值方法setValue:forKey:
的流程圖
當我們呼叫setValue:forKey:
設定屬性時
- 優先呼叫
setKey、_setKey
方法 - 沒找到上面的方法, 則呼叫
accessInstanceVariablesDirectly
方法,該方法預設返回YES,會按照_key,_iskey,key,iskey
的順序查詢成員變數 - 如果按照上面的順序都搜尋不到成員變數,則會呼叫
setValue:forUndefinedKey:
,並丟擲異常
注意查詢過程中不管這些方法、成員變數是私有的還是公共的都能正確設定
瞭解完設定屬性,再來看看valueForkey:
方法取值的流程圖
可以看到,整個流程和設定屬性值的步驟是一模一樣的,只不過查詢的方法不一樣,取值的時候
- 優先按照
getKey、key、isKey、_key
的順序查詢方法 - 找不到上面的方法,會按照
_key,_iskey,key,iskey
的順序查詢成員變數 - 如果按照上面的順序都搜尋不到成員變數,則會呼叫
setValue:forUndefinedKey:
,並丟擲異常
注意查詢過程中不管這些方法、成員變數是私有的還是公共的都能正確讀取到值
KVO(Key-Value Observing)鍵值觀察
KVO是一種觀察者模式的衍生,用於監聽某個物件屬性值的改變。
簡單來說KVO可以通過監聽物件屬性的key
,來獲得value
在Objective-C中要實現KVO則必須實現NSKeyValueObServing協議,而NSObject已經實現了改協議,因此對於所有繼承了NSObject的型別,也就是說Object-C中幾乎所有的物件都支援KVO操作,常用的KVO操作方法如下:
/**
註冊觀察者
observer:觀察者,也就是KVO通知的訂閱者。
keyPath:描述將要觀察的屬性,相當於被觀察者。
options:KVO的一些屬性配置。
NSKeyValueObservingOptionNew:change字典包括改變後的值
NSKeyValueObservingOptionOld:change字典包括改變前的值
NSKeyValueObservingOptionInitial:註冊後立刻觸發KVO通知
NSKeyValueObservingOptionPrior:值改變前是否也要通知(這個key決定了是否在改變前改變後通知兩次)
context: 上下文,這個會傳遞到訂閱著的函式中,用來區分訊息。
*/
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
// 移除觀察者
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(nullable void *)context
// 移除觀察者
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath;
// 監聽回撥方法, change 這個字典儲存了變更資訊,具體是哪些資訊取決於註冊觀察者時的options
- (void)observeValueForKeyPath:(NSString *)keyPath
ofObject:(id)object
change:(NSDictionary *)change
context:(void *)context
複製程式碼
使用KVO的整個流程就是
- 註冊觀察者
addObserver: forKeyPath: options: context:
- 實現回撥方法
observeValueForKeyPath: ofObject: change: context:
- 在合適的時機,移除觀察者
removeObserver: forKeyPath
、removeObserver: forKeyPath: context:
簡單的應用程式碼表現為
KLPerson.h
@interface KLPerson : NSObject
// 公開屬性
@property (nonatomic,readwrite,copy) NSString *name;
@end
KLPerson.m
@implementation KLPerson {
// 私有屬性
int _age;
}
複製程式碼
#pragma mark --- Lifecycle
- (void)dealloc {
// 移除觀察者
[self.person removeObserver:self forKeyPath:@"name"];
[self.person removeObserver:self forKeyPath:@"age"];
}
- (void)viewDidLoad {
[super viewDidLoad];
// 監聽 Person 的公開屬性
[self.person addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];
// 監聽 Person 的私有屬性
[self.person addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];
self.person.name = @"小紅";
// 使用KVC對私有屬性賦值
[self.person setValue:@10 forKey:@"age"];
}
#pragma mark --- OverwriteSuperClass
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context{
// 通常為以下的寫法
if (object == self.person && [keyPath isEqualToString:@"name"]) {
// 做些什麼...
} else if (object == self.person && [keyPath isEqualToString:@"age"]) {
// 做些什麼...
} else {
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
}
NSLog(@"監聽到%@的%@改變了%@",object,keyPath,change);
}
輸出的結果為:
監聽到<KLPerson: 0x600003008560>的name改變了{
kind = 1;
new = "小紅";
old = "<null>";
},
監聽到<KLPerson: 0x600003008560>的age改變了{
kind = 1;
new = 10;
old = 0;
}
複製程式碼
從輸出的Log,可以看得出來,當被觀察者物件的屬性值改變時,觀察者可以通過 observeValueForKeyPath: ofObject: change: context:
回撥方法獲取到改變的值,去搞些事情~
使用KVC對私有屬性賦值時,也會觸發回撥~
如果想要了解更多 KVO底層原理的實現,可以看下這篇文章,很詳細iOS底層原理總結 - 探尋KVO本質
Q & A
-
Q :iOS用什麼方式實現對一個物件的KVO?(KVO的本質是什麼?)
-
A :當一個物件使用了KVO監聽,iOS系統會修改這個物件的isa指標,改為指向一個全新的通過Runtime動態建立的子類 NSKVONotifying_物件的類,子類擁有自己的set方法實現,set方法實現內部會順序呼叫
-
willChangeValueForKey:
方法 - 原來的setter方法實現
-
didChangeValueForKey:
方法
而
didChangeValueForKey:
方法內部又會呼叫監聽器的observeValueForKeyPath:ofObject:change:context:
監聽方法。 -
-
Q : 如何手動觸發KVO?
-
A :被監聽的屬性的值被修改時,就會自動觸發KVO。如果想要手動觸發KVO,則需要我們自己呼叫
willChangeValueForKey:
和didChangeValueForKey:
方法,即可在不改變屬性值的情況下手動觸發KVO,並且這兩個方法缺一不可。