原文:http://gaoyong.diandian.com/post/2012-08-08/40031793795
KVO (Key Value Observing)是ios里面一种特别方便的机制用于“捕捉”对象属性的变化。在概念理解上,是设计模式里面观察者模式的一种实践。
拿一个具体的例子来讲:
有一个数据对象EmployeeData,该对象有一个属性salary
有一个ViewController 用于显示对象EmployeeData的属性salary的值
当salary的值发生变化的时候,ViewController如何显示变化后的新值。
方案一:
“可以在EmployeeData类弱引用ViewController类,然后在EmployeeData数据发生变化时,调
用ViewController类的回调函数。这个方法虽然能达到目的,但是会破坏EmployeeData的完整
性,一个负责数据管理的类,不应当依赖另一个负责视图控制的类;换句话说,EmployeeData
类不应该知道关于ViewController类的任何事情,甚至不需要知道其从在”
方案二:
使用ios提供的消息中心(NSNotificationCenter)。在此EmployeeData为消息生产者,ViewController为消息消费者。当salary数据发生变化时发送一个消息既可。
ViewController接收到salary数据变化的通知做相对应的业务处理。不足之处如同方案一,EmployeeData对象的salary每次变动都需要发送“通知”。这项工作对EmployeeData自身来讲毫无意义。
方案三:
在ViewController中将自身(self)设置为EmployeeData salary属性值变化的观察者。当salary值发生变化时,执行一个回调方法。这样对“EmployeeData”来讲不用关心除自身业务以
外的事情。避免了方案一和方案二的瑕疵。对ViewController来讲关注谁的变化,注册自己为其的观察者既可。间接轻便。
下面是一些核心的代码。
KVO (Key Value Observing)是ios里面一种特别方便的机制用于“捕捉”对象属性的变化。在概念理解上,是设计模式里面观察者模式的一种实践。
拿一个具体的例子来讲:
有一个数据对象EmployeeData,该对象有一个属性salary
有一个ViewController 用于显示对象EmployeeData的属性salary的值
当salary的值发生变化的时候,ViewController如何显示变化后的新值。
方案一:
“可以在EmployeeData类弱引用ViewController类,然后在EmployeeData数据发生变化时,调
用ViewController类的回调函数。这个方法虽然能达到目的,但是会破坏EmployeeData的完整
性,一个负责数据管理的类,不应当依赖另一个负责视图控制的类;换句话说,EmployeeData
类不应该知道关于ViewController类的任何事情,甚至不需要知道其从在”
方案二:
使用ios提供的消息中心(NSNotificationCenter)。在此EmployeeData为消息生产者,ViewController为消息消费者。当salary数据发生变化时发送一个消息既可。
ViewController接收到salary数据变化的通知做相对应的业务处理。不足之处如同方案一,EmployeeData对象的salary每次变动都需要发送“通知”。这项工作对EmployeeData自身来讲毫无意义。
方案三:
在ViewController中将自身(self)设置为EmployeeData salary属性值变化的观察者。当salary值发生变化时,执行一个回调方法。这样对“EmployeeData”来讲不用关心除自身业务以
外的事情。避免了方案一和方案二的瑕疵。对ViewController来讲关注谁的变化,注册自己为其的观察者既可。间接轻便。
下面是一些核心的代码。
- //
- // EmployeeData.h
- // KVOExample
- //
- // Created by gaoyong on 12-8-8.
- // Copyright (c) 2012年 gaoyong. All rights reserved.
- //
- #import <Foundation/Foundation.h>
- @interface EmployeeData : NSObject {
- NSString *salary;
- }
- @property(nonatomic,retain) NSString *salary;
- @end
- //
- // EmployeeData.m
- // KVOExample
- //
- // Created by gaoyong on 12-8-8.
- // Copyright (c) 2012年 gaoyong. All rights reserved.
- //
- #import "EmployeeData.h"
- @implementation EmployeeData
- @synthesize salary;
- @end
- //
- // ViewController.h
- // KVOExample
- //
- // Created by gaoyong on 12-8-8.
- // Copyright (c) 2012年 gaoyong. All rights reserved.
- //
- #import <UIKit/UIKit.h>
- @interface ViewController : UIViewController {
- UILabel *salary;
- }
- @property(nonatomic,retain) IBOutlet UILabel *salary;
- @end
- //
- // ViewController.m
- // KVOExample
- //
- // Created by gaoyong on 12-8-8.
- // Copyright (c) 2012年 gaoyong. All rights reserved.
- //
- #import "ViewController.h"
- #import "EmployeeData.h"
- @interface ViewController () {
- EmployeeData *employeeData;
- }
- @end
- @implementation ViewController
- @synthesize salary;
- - (void)viewDidLoad
- {
- [super viewDidLoad];
- // Do any additional setup after loading the view, typically from a nib.
- employeeData = [[EmployeeData alloc] init];
- [employeeData addObserver:self forKeyPath:@"salary" options:NSKeyValueObservingOptionNew context:nil];
- }
- -(void)viewDidAppear:(BOOL)animated {
- //employeeData.salary = @"20";
- //salary.text = employeeData.salary;
- employeeData.salary = @"20";
- employeeData.salary = @"200";
- employeeData.salary = @"2000";
- employeeData.salary = @"20000";
- employeeData.salary = @"200000";
- }
- - (void)viewDidUnload
- {
- [super viewDidUnload];
- // Release any retained subviews of the main view.
- }
- - (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
- {
- return (interfaceOrientation != UIInterfaceOrientationPortraitUpsideDown);
- }
- -(void) observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
- NSLog(@"observeValueForKeyPath is run"); // 美妙在这里:这一行会打印5次。
- if (object == employeeData && [keyPath isEqualToString:@"salary"]) {
- self.salary.text = employeeData.salary;
- }
- }
- -(void) dealloc {
- [employeeData removeObserver:self forKeyPath:@"salary"];
- }
- @end