1. 程式人生 > IOS開發 >【iOS面試糧食】OC語言—KVC、KVO

【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:的流程圖

DC44211B-F516-4C05-B5C1-C192B91D0CFD.png

當我們呼叫setValue:forKey:設定屬性時

  • 優先呼叫setKey、_setKey方法
  • 沒找到上面的方法, 則呼叫accessInstanceVariablesDirectly方法,該方法預設返回YES,會按照_key,_iskey,key,iskey的順序查詢成員變數
  • 如果按照上面的順序都搜尋不到成員變數,則會呼叫setValue:forUndefinedKey:,並丟擲異常

注意查詢過程中不管這些方法、成員變數是私有的還是公共的都能正確設定

瞭解完設定屬性,再來看看valueForkey:方法取值的流程圖

1B65FE64-5E09-460D-B522-760A4A81AB0F.png

可以看到,整個流程和設定屬性值的步驟是一模一樣的,只不過查詢的方法不一樣,取值的時候

  • 優先按照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: forKeyPathremoveObserver: 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

  1. Q :iOS用什麼方式實現對一個物件的KVO?(KVO的本質是什麼?)

  2. A :當一個物件使用了KVO監聽,iOS系統會修改這個物件的isa指標,改為指向一個全新的通過Runtime動態建立的子類 NSKVONotifying_物件的類,子類擁有自己的set方法實現,set方法實現內部會順序呼叫

    • willChangeValueForKey:方法
    • 原來的setter方法實現
    • didChangeValueForKey:方法

    didChangeValueForKey:方法內部又會呼叫監聽器的observeValueForKeyPath:ofObject:change:context:監聽方法。

  3. Q : 如何手動觸發KVO?

  4. A :被監聽的屬性的值被修改時,就會自動觸發KVO。如果想要手動觸發KVO,則需要我們自己呼叫willChangeValueForKey:didChangeValueForKey:方法,即可在不改變屬性值的情況下手動觸發KVO,並且這兩個方法缺一不可。

參考資料

iOS底層原理總結 - 探尋KVO本質

iOS開發系列--Objective-C之KVC、KVO

iOS開發技巧系列---詳解KVC(我告訴你KVC的一切)