[KVOExample]ios里如何使用KVO

转自: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来讲关注谁的变化,注册自己为其的观察者既可。间接轻便。

 

下面是一些核心的代码。

 

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