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
#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
#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