天天看点

KVO(键值监听)详解

iOS通常会把应用程序组件分开成数据模型组件和视图组件,其中数据模型负责维护应用程序的状态数据,而视图组件负责显示数据模型组件内部的状态数据。对于上面的设计模式,如果程序存在的需求是:在数据模型组件的状态数据发生改变时,试图组件能动态的更新自己,及时显示数据模型组件更新后的数据。为了解决上面的需求,及时显示数据模型组件更新后的数据:

1、我们可以用通知中心,但是对于数据模型组件和视图模型组件之间都需要与iOS的消息中心耦合,而且每当数据模型组件的内部状态发生改变时,都需要向iOS的消息中心发送消息,这也是非常令人悲哀的事情。

2、那么如果我们用KVO那么就是很好的解决方案了,(键值监听机制)

下方方法:

addObserver: forKeyPath: options: context: 注册一个监听器用于监听指定的key路径

removeObserver: forKeyPath: 为key路径删除指定的监听器

removeObserver: forKeyPath: context:为key路径删除指定的监听器.只是多了一个context参数

 当数据模型组件的key路径对应属性值发生改变时,作为监听器的视图组件将会被激发,假发是就会回调监听器自身的监听方法,该监听方法如下:

-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context

{

}

由此可见,作为监听器的试图组件需要重写上方这个方法,重写该方法是就可以得到最新修改的数据,从而使用最新的数据来更新视图组件的显示。

从上面介绍不难看出,KVO编程的步骤如下:

1、为被监听对象(通常是数据模型组件)注册监听器

2、重写监听器的-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context方法。

下方用一个简单地类模拟视图组件,下方代码仔细看即可:

程序代码:LYXItem.h

#import <Foundation/Foundation.h>

@interface LYXItem : NSObject
@property(nonatomic,copy) NSString *name;
@property(nonatomic,copy) NSString *price;
@end
           

程序代码:LYXItemView.h

//  Created by lyx on 16/2/11.
//  Copyright (c) 2016年 李云祥. All rights reserved.
//

#import <Foundation/Foundation.h>
#import "LYXItem.h"

@interface LYXItemView : NSObject

//使用@property定义两个属性
@property(nonatomic,weak) LYXItem * item;
-(void)showItemInfo;

@end
           

程序代码:LYXItemView.m

#import "LYXItemView.h"

@implementation LYXItemView
@synthesize item = _item;

-(void)showItemInfo
{
  
    NSLog(@"item名为:%@,价格为%@",self.item.name,self.item.price);
}
//自定义setItem方法
-(void)setItem:(LYXItem *)item
{
    self->_item = item;
    //为item添加监听器,监听item的name的属性的变化
    [self.item addObserver:self forKeyPath:@"name" options:NSKeyValueObservingOptionNew context:nil];
    //为item添加监听器,监听item的price属性的变化
    [self.item addObserver:self forKeyPath:@"price" options:NSKeyValueObservingOptionNew context:nil];
    
}
//重写该方法,当被监听的数据模型发生改变时,就会回调监听器的该方法
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
    NSLog(@"-------observeValueForKeyPath-----方法被调用");
    NSLog(@"被修改的keyPath为:%@",keyPath);
    NSLog(@"被修改的对象为:%@",object);
    NSLog(@"新被修改的属性值是:%@",[change objectForKey:@"new"]);
    NSLog(@"被修改的上下文是:%@",context);
}
-(void)dealloc
{
    //删除监听器
    [self.item removeObserver:self forKeyPath:@"name"];
    [self.item removeObserver:self forKeyPath:@"price"];
    
}
@end
           

程序代码:main.m

#import <Foundation/Foundation.h>
#import "LYXItemView.h"
#import "LYXItem.h"

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        // insert code here...
        //创建item对象
        LYXItem *item = [[LYXItem alloc]init];
        item.name = @"IOS";
        item.price = @"10";
        
        LYXItemView *itemView = [[LYXItemView alloc]init];
        //将itemView的item属性设为item
        itemView.item = item;
        [itemView showItemInfo];
        //再次更改item的属性,将会激发监听器的方法
        item.name = @"安卓";
        item.price = @"100";
        
        
    }
    return 0;
}
           

输出下方代码:

2016-02-11 22:51:49.767 KVO操练[502:17531] item名为:IOS,价格为10
2016-02-11 22:51:49.768 KVO操练[502:17531] -------observeValueForKeyPath-----方法被调用
2016-02-11 22:51:49.768 KVO操练[502:17531] 被修改的keyPath为:name
2016-02-11 22:51:49.769 KVO操练[502:17531] 被修改的对象为:<LYXItem: 0x100304830>
2016-02-11 22:51:49.769 KVO操练[502:17531] 新被修改的属性值是:安卓
2016-02-11 22:51:49.769 KVO操练[502:17531] 被修改的上下文是:(null)
2016-02-11 22:51:49.769 KVO操练[502:17531] -------observeValueForKeyPath-----方法被调用
2016-02-11 22:51:49.769 KVO操练[502:17531] 被修改的keyPath为:price
2016-02-11 22:51:49.769 KVO操练[502:17531] 被修改的对象为:<LYXItem: 0x100304830>
2016-02-11 22:51:49.770 KVO操练[502:17531] 新被修改的属性值是:100
2016-02-11 22:51:49.770 KVO操练[502:17531] 被修改的上下文是:(null)
Program ended with exit code: 0           

可以看出,当使用KVO后,此时的程序当被监听的数据模型组件LYXItem的属性发生改变,监听器的监听方法就会被激发。要是在视图控制器上,那么这里的UI就会进行改变。

KVO(键值监听)详解