Objective-C语法之KVO的使用

来源:互联网 发布:java ee是干什么的 编辑:程序博客网 时间:2024/06/11 05:01

http://www.189works.com/article-84627-1.html

简介:

上篇我们讲到了KVC,这篇我们学习KVO,全名为:Key Value Observing,直译为:基于键值的观察者。

那它有什么用呢?KVO主要用于视图交互方面,比如界面的某些数据变化了,界面的显示也跟着需要变化,那就要建立数据和界面的关联。

ObjC中提供的KVO就是解决这种问题的。以下用显示页面观察学生的课程名称变化的例子来说明KVO的使用。

学生类命名为:Student,页面类是:PageView.

上图来自苹果官网:图中的BankObject好比PageView,PersonObject好比Student,

PageView观察Student的变化。

1、添加Student学生类。

.h

#import @interface Student : NSObject{    NSString *name;    NSString *courseName;}-(void)changeCourseName:(NSString*) newCourseName;@end

类中有name,和课程名称courseName,添加一个可以改变课程名称的方法changeCourseName。一会用来做对比,看直接改变课程名称时会不会有回调。

实现文件.m

#import "Student.h"@implementation Student-(void)changeCourseName:(NSString*) newCourseName{    courseName = newCourseName;}@end

实现类把方法实现了。

2、页面类实现

.h文件

#import @class Student;@interface PageView : NSObject{    Student *student;}-(id)init:(Student*)initStudent;@end

.m文件

#import "PageView.h"#import "Student.h"@implementation PageView-(id)init:(Student*)initStudent{    if (self = [super init]) {        student = initStudent;        [student addObserver:self                   forKeyPath:@"courseName"                      options:NSKeyValueObservingOptionOld                            |NSKeyValueObservingOptionNew context:nil];    }    return self;}- (void) dealloc{     [student removeObserver:self forKeyPath:@"courseName" context:nil];    [super dealloc]; }-(void)observeValueForKeyPath:(NSString *)keyPath                      ofObject:(id)object                        change:(NSDictionary *)change                       context:(void *)context{    if ([keyPath isEqual:@"courseName"]) {        NSLog(@"PageView课程被改变了");        NSLog(@"PageView新课程是:%@ 老课程是:%@", [change objectForKey:@"new"],[change objectForKey:@"old"]);    }}@end

init初始化时,向student实例添加观察者,在释放的时候移除观察者。

3、实现观察

在main函数中

#import "Student.h"#import "Course.h"#import "PageView.h"int main(int argc, const char * argv[]){    @autoreleasepool {        Student *student = [[[Student alloc]init]autorelease];        [student changeCourseName:@"数学课"];        NSLog(@"初始值:%@", [student valueForKey:@"courseName"]);                //创建页面实例        PageView *pageview = [[[PageView alloc]init:student]autorelease];                [student setValue:@"化学课" forKey:@"courseName"];          }    return 0;}

新建一个student的实例,设置他的课程是数学课,然后创建页面类的时候,用student初始化。这是页面类已经观察着学生的课程了。

 

再给课程设置新的值为化学课。这时候运行打印结果:

 

2012-07-24 16:29:21.561 objectiveC[2192:403] 初始值:数学课

2012-07-24 16:29:21.565 objectiveC[2192:403] PageView课程被改变了

2012-07-24 16:29:21.566 objectiveC[2192:403] PageView新课程是:化学课 老课程是:数学课

可以看到Pageview类中的回调被调用,Pageview接收到学生课程数据更新的信息。

4、直接改变课程信息对比

#import "Student.h"#import "Course.h"#import "PageView.h"int main(int argc, const char * argv[]){    @autoreleasepool {        Student *student = [[[Student alloc]init]autorelease];        [student changeCourseName:@"数学课"];        NSLog(@"初始值:%@", [student valueForKey:@"courseName"]);                //创建页面实例        PageView *pageview = [[[PageView alloc]init:student]autorelease];                [student setValue:@"化学课" forKey:@"courseName"];        [student changeCourseName:@"英语课"];        NSLog(@"直接改变的课程为:%@", [student valueForKey:@"courseName"]);    }    return 0;}

直接调用changeCourseName方法改变课程,打印结果:

2012-07-24 16:32:06.230 objectiveC[2240:403] 初始值:数学课

2012-07-24 16:32:06.237 objectiveC[2240:403] PageView课程被改变了

2012-07-24 16:32:06.238 objectiveC[2240:403] PageView新课程是:化学课 老课程是:数学课

2012-07-24 16:32:06.239 objectiveC[2240:403] 直接改变的课程为:英语课

可以看到,这时Pageview的回调没被调用到。说明只有通过键值编码(KVC)改变的值,才会回调观察者注册的方法。

这里是苹果官网的关于KVO的文档,英文好的朋友可以看看:

https://developer.apple.com/library/mac/#documentation/Cocoa/Conceptual/KeyValueObserving/KeyValueObserving.html#//apple_ref/doc/uid/10000177-BCICJDHA 



-----------------------------------------------------

KVO这东西,早就听说过,也看过很多相关的文章,知道是用来监听一个类里面的一个值改变,然后执行点儿运作的,只是看过的文章都讲的很理论很复杂,以至于只是知道个大概:设置皮肤统一换背景的时候用它。终于我也遇到了这样的一个场景:用户登陆成功以后,需要获取用户状态,根据不同的用户状态要展示不同的界面,而且还有一点:用户状态是变的。没想到用KVO的方式很优雅的就实现了这个东东。因为我取到用户的数据是保存在NSUserDefault里的,所以基于KVO我只需要监听NSUserDefault里的这个值的改变就可以了

[NDSUD addObserver:self
        forKeyPath:@"CurentUserOnLineInformation"
           options:NSKeyValueObservingOptionNew
           context:nil];

上面的代码可以搁在init里,也可以搁到ViewDidLoad里。然后还要在这个文件里实现这个方法

- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary *)change
                       context:(void *)context{
  if ([keyPath isEqualToString:@"CurentUserOnLineInformation"]) {
    NSDictionary *_NewValue = [change valueForKey:@"new"];
    if ([_NewValue isKindOfClass:[NSDictionary class]]) {
      //用户登陆或者续期了
    }
    else{
      //用户登出
    }
  }
}

没想到KVO这么简单就实现了,就像NSUserDefault一样,很方便,很好使。
扩展一下,两个页面间传值现在是不是又多了一个选择?
再扩展一下:像这样在一个地方定义一个东西,再到另一个地方去实现一个方法,总是不那么易读易维护,So,还是上Block吧:)
搜索了一下,发现基于KVO的block,实现很多,试了一遍,推荐使用这个:kvo-block-binding

[NDSUD addObserverForKeyPath:@"CurentUserOnLineInformation"
                       owner:self
                     options:NSKeyValueObservingOptionNew
                       block:^(id observed, NSDictionary *change) {
                         NSLog(@"%@",change);

                       }];

这样就简单易读多了。移除的方法:

[NDSUD removeAllBlockBasedObserversForOwner:self];

为了让你的工程更易于维护,实现更优雅,建议多用Block,多用KVO!

转载请注明: 转自Rainbird的个人博客
   本文链接: Hello KVO,相见恨晚!


原创粉丝点击