當你不愛我的時候,請告訴我(KVC/KVO/NSNotification)

KVC KVO NSNotification


一、KVC

  • 基本概念

    1. 它是一種可以直接通過字符串類型的屬性名(key)來訪問某個類屬性的機制。而不是通過調用Setter、Getter方法訪問。

    2. 所有NSObject對象都可以使用KVC。

    3. KVC既支持帶有對象值的屬性,也支持基本數(shù)據(jù)類型和結構。基本數(shù)據(jù)類型會被自動封裝和解裝。

  • 基本使用

    1. 通過 setValue: forKey: 設置對象的值。

    2. 通過 valueForKey: 獲得對象的值。

    3. 通過 setValue: forKeyPath: 設置指定路徑的對象值。

    4. 通過 valueForKeyPath: 獲得指定路徑的對象值。

示例代碼如下:

```objc    
//Person.h

@interface Person : NSObject

@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) NSInteger age;
@property (nonatomic, strong) Person *girlFriend;
@property (nonatomic, copy) NSArray *books;

@end
```     

```objc    
- (void)viewDidLoad {
    [super viewDidLoad];

    Person *person = [[Person alloc] init];
    
    NSLog(@"1: %@ - %ld", person.name, person.age);
    
    [person setValue:@"Bourne" forKey:@"name"];
    [person setValue:@21 forKey:@"age"];
    
    NSLog(@"2: %@ - %ld", [person valueForKey:@"name"], person.age);
    
    Person *girl = [[Person alloc] init];
    [person setValue:girl forKey:@"girlFriend"];
    
    NSLog(@"3: %@ - %ld", [person valueForKeyPath:@"girlFriend.name"], person.girlFriend.age);
    
    [person setValue:@"willYou?" forKeyPath:@"girlFriend.name"];
    [person setValue:@20 forKeyPath:@"girlFriend.age"];
    
    NSLog(@"4: %@ - %ld", person.girlFriend.name, person.girlFriend.age);
}
```        
 
**打印輸出**       

```
2015-04-06 16:49:19.192 KVC-KVO-test[5685:7429728] 1: (null) - 0
2015-04-06 16:49:19.193 KVC-KVO-test[5685:7429728] 2: Bourne - 21
2015-04-06 16:49:19.194 KVC-KVO-test[5685:7429728] 3: (null) - 0
2015-04-06 16:49:19.194 KVC-KVO-test[5685:7429728] 4: willYou? - 20
```           
  • 一對多關系

    1. 如果向NSArray等集合類型數(shù)據(jù)所包含的對象發(fā)送請求消息,它會查詢數(shù)組中每個對象來查找這個鍵值,并返回一個打包后的數(shù)組。

    2. 如果想要向數(shù)組等集合類型的數(shù)據(jù)所包含的對象發(fā)送賦值消息,它也會查詢數(shù)組中每個對象來查找這個鍵值,并賦同一個值。注意:不要企圖發(fā)送一個數(shù)組讓她為找到的對象按順序賦值,這樣會報錯!

    示例代碼如下

    //Book.h
    
    @interface Book : NSObject
    
    @property (nonatomic, assign) NSInteger price;
    
    @end
    
    - (void)setAndGetBooks {
        Person *person = [[Person alloc] init];
        
        Book *book1 = [[Book alloc] init];
        Book *book2 = [[Book alloc] init];
        Book *book3 = [[Book alloc] init];
        
        NSArray *myBooks = @[book1, book2, book3];
        
        [person setValue:myBooks forKey:@"books"];
        
        NSLog(@"1: %@", [person valueForKeyPath:@"books.price"]);
        
        [person setValue:@1 forKeyPath:@"books.price"];
        
        NSLog(@"2: %@", [person valueForKeyPath:@"books.price"]);
    }
    

    打印輸出

        2015-04-06 17:12:48.256 KVC-KVO-test[5886:7519923] 1: (
        0,
        0,
        0
    )
    2015-04-06 17:12:48.257 KVC-KVO-test[5886:7519923] 2: (
        1,
        1,
        1
    )
    

二、KVO

  • 基本概念

    1. KVO: Key-Value Observing,它提供一種機制,當指定的對象的屬性被修改后,則對象就會接受到通知。就是每次指定的被觀察的對象的屬性被修改后,KVO就會自動通知相應的觀察者。屬于一種非正式的 Protocol。

    2. 可以觀察任意屬性,包括基本數(shù)據(jù)類型,對一或對多關系。對多關系的觀察者將會被告知發(fā)生變化的類型,也即使任意發(fā)生變化的對象。

  • 使用方法

    1. 注冊觀察者,指定被觀察者的屬性。

    2. 實現(xiàn)回調方法,接收變更通知。

    3. 移除觀察者身份。

  • 注冊觀察者

    為了正確接收屬性的變更通知,觀察者必須首先發(fā)送一個 addObserver:forKeyPath:options:context: 消息給被觀察者。用以傳送需要觀察的對象和需要觀察的屬性的關鍵路徑,選型參數(shù)指定了發(fā)生變更時提供給觀察者的信息,使用 NSKeyValueObservingOptionNew 可以在回調方法中獲得舊值,使用 NSKeyValueObservingOptionOld 可以在回調方法中獲得新值。

    示例

    [person addObserver:self forKeyPath:@"books.price" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:NULL];
    
  • 回調方法

    當監(jiān)聽屬性發(fā)生變化時,堅挺著將會收到一條 observeValueForKeyPath: ofObject: change: context: 消息。觸發(fā)觀察的對象、鍵路徑、包含變化細節(jié)的字典都會傳給觀察者。

        -(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {  
        if([keyPath isEqualToString:@"books.price"]) {  
            NSLog("通知我啦!");  
        }  
    } 
    
  • 移除觀察者

    你可以發(fā)送一條包含觀察者,鍵路徑的方法 `` 給被觀察的對象,來移除這個觀察。
    注意:注冊了觀察者就需要手動移除觀察者!

    [person removeObserver:self forKeyPath:@"books.price"];
    

三、NSNotification

  • 概述

    與KVO不同的是,KVO在屬性上通過 K-V 發(fā)生改變時,自動調用 observeValueForKeyPath: ofObject: change: context: 方法,而 NSNotification 在需要的時候自行發(fā)送通知才調用,且方法自定義。

  • 使用方法

    1. NSNotificationCenter 注冊觀察者時指定事件(以字符串命名),及該事件觸發(fā)時該執(zhí)行的 Selector 或 Block

    2. NSNotificationCenter 在某個時機激發(fā)事件(以字符串命名)

      注意:編譯器根據(jù)這個事件的名字找到之前注冊過這個名字的觀察者,所以這個名字一般設置成靜態(tài)屬性,保證兩次使用是同一個字符串!

    3. 觀察者在收到感興趣的事件時,執(zhí)行相應的 Selector 或 Block

  • 注冊觀察者

    使用默認的通知中心,指定觀察者、需要接收的通知名稱、接收通知時執(zhí)行的方法,最后一個參數(shù)是表示會對哪個發(fā)送者對象發(fā)出的事件作出響應,nil 時表示接受所有發(fā)送者的事件。

    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(action:) name:@"NOTIFICATION_NAME" object:nil];
    
  • 激發(fā)觀察者

    需要指定通知的名稱,編譯器根據(jù)這個事件的名字找到之前注冊過這個名字的觀察者,還可以指定傳給觀察者的對象。

    NSObject *test = [[NSObject alloc] init]; 
    [[NSNotificationCenter defaultCenter] postNotificationName:@"NOTIFICATION_NAME" object:test];     
    
    //或者以下的方式:   
    
    NSNotification *notification = [NSNotification  notificationWithName:@"NOTIFICATION_NAME" object:test];      
    [[NSNotificationCenter defaultCenter] postNotification:notification];
    
  • 執(zhí)行事件

    觀察者接受到通知后就會執(zhí)行注冊時指定的方法,并獲得被觀察者傳回的對象。

    - (void)action:(NSNotification *)notification {
        NSLog("通知我啦!");
    } 
    

聲明

  1. 以上內容屬于本人整理的筆記,如有錯誤的地方希望能告訴我,大家共同進步。

  2. 以上內容有些段落或語句可能是本人從其他地方Copy而來,如有侵權,請及時告訴我。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容