KVC和KVO总结

来源:互联网 发布:ubuntu gnome16.04美化 编辑:程序博客网 时间:2024/06/03 19:33

1、KVC,即是指 NSKeyValueCoding,一个非正式的Protocol,提供一种机制来间接访问对象的属性。而不是通过调用Setter、Getter方法访问。KVO 就是基于 KVC 实现的关键技术之一。

Demo:

@interface myPerson : NSObject{        NSString*_name;        int      _age;        int      _height;        int      _weight;} @end@interface testViewController :UIViewController @property (nonatomic, retain) myPerson*testPerson; @end//可以不通过set和get读取和写入数据(重点)- (void)testKVC{    testPerson = [[myPerson alloc] init];            NSLog(@"testPerson‘s init height =%@", [testPerson valueForKey:@"height"]);        [testPerson setValue:[NSNumber numberWithInt:168]forKey:@"height"];      NSLog(@"testPerson‘s height = %@", [testPerson valueForKey:@"height"]);}

第一段代码是定义了一个myPerson的类,这个类有一个_height的属性,但是没有提供任何getter/setter的访问方法。同时在testViewController这个类里面有一个myPerson的对象指针。

   当myPerson实例化后,常规来说是无法访问这个对象的_height属性的,不过通过KVC我们做到了,代码就是testKVC这个函数。   运行之后打印值就是:    2015-3-13 11:16:21.970 test[408:c07] testPerson‘s init height = 0     2015-3-13 11:16:21.971 test[408:c07] testPerson‘s height = 168    这就说明确实读写了_height属性。

KVC的常用方法:
- (id)valueForKey:(NSString )key; -(void)setValue:(id)value forKey:(NSString )key;
valueForKey的方法根据key的值读取对象的属性,setValue:forKey:是根据key的值来写对象的属性。

注意:

(1). key的值必须正确,如果拼写错误,会出现异常

(2). 当key的值是没有定义的,valueForUndefinedKey:这个方法会被调用,如果你自己写了这个方法,key的值出错就会调用到这里来

(3). 因为类key反复嵌套,所以有个keyPath的概念,keyPath就是用.号来把一个一个key链接起来,这样就可以根据这个路径访问下去

(4). NSArray/NSSet等都支持KVC


2、KVO的是KeyValue Observe的缩写,中文是键值观察。这是一个典型的观察者模式,观察者在键值改变时会得到通知。iOS中有个Notification的机制,也可以获得通知,但这个机制需要有个Center,相比之下KVO更加简洁而直接。

  KVO的使用也很简单,就是简单的3步。  1.注册需要观察的对象的属性addObserver:forKeyPath:options:context:  2.实现observeValueForKeyPath:ofObject:change:context:方法,这个方法当观察的属性变化时会自动调用  3.取消注册观察removeObserver:forKeyPath:context:

Demo:

@interface myPerson : NSObject  {  NSString *_name;  int      _age;  int      _height;  int      _weight;  }  @end  @interface testViewController : UIViewController  @property (nonatomic, retain) myPerson *testPerson;  - (IBAction)onBtnTest:(id)sender;  @end  - (void)testKVO  {      testPerson = [[myPerson alloc] init];      [testPerson addObserver:self forKeyPath:@"height" options:NSKeyValueObservingOptionNew context:nil];  }  - (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context  {   if ([keyPath isEqualToString:@"height"]) {    NSLog(@"Height is changed! new=%@", [change valueForKey:NSKeyValueChangeNewKey]);  } else {      [super observeValueForKeyPath:keyPath ofObject:object change:change context:context];      }  }  - (IBAction)onBtnTest:(id)sender {      int h = [[testPerson valueForKey:@"height"] intValue];          [testPerson setValue:@(35) forKey:@"height"];      NSLog(@"person_height=%@",[testPerson valueForKey:@"height"]);  }  - (void)dealloc  {      [testPerson removeObserver:self forKeyPath:@"height" context:nil];      [super dealloc];  }  

第一段代码声明了myPerson类,里面有个_height的属性。在testViewController有一个testPerson的对象指针。
在testKVO这个方法里面,我们注册了testPerson这个对象height属性的观察,这样当testPerson的height属性变化时, 会得到通知。在这个方法中还通过NSKeyValueObservingOptionNew这个参数要求把新值在dictionary中传递过来。
重写了observeValueForKeyPath:ofObject:change:context:方法,这个方法里的change这个NSDictionary对象包含了相应的值。

需要强调的是KVO的回调要被调用,属性必须是通过KVC的方法来修改的,如果是调用类的其他方法来修改属性,这个观察者是不会得到通知的。

KVO实现原理:

    当某个类的对象第一次被观察时,系统就会在运行期动态地创建该类的一个派生类,在这个派生类中重写基类中任何被观察属性的 setter 方法。 派生类在被重写的 setter 方法实现真正的通知机制,就如前面手动实现键值观察那样。这么做是基于设置属性会调用 setter 方法,而通过重写就获得了 KVO 需要的通知机制。当然前提是要通过遵循 KVO 的属性设置方式来变更属性值,如果仅是直接修改属性对应的成员变量,是无法实现 KVO 的。 同时派生类还重写了 class 方法以“欺骗”外部调用者它就是起初的那个类。然后系统将这个对象的 isa 指针指向这个新诞生的派生类,因此这个对象就成为该派生类的对象了,因而在该对象上对 setter 的调用就会调用重写的 setter,从而激活键值通知机制。此外,派生类还重写了 dealloc 方法来释放资源。    新类会重写对应的set方法,是为了在set方法中增加另外两个方法的调用:- (void)willChangeValueForKey:(NSString *)key ; - (void)didChangeValueForKey:(NSString *)key ;    其中,didChangeValueForKey:方法负责调用:- (void)observeValueForKeyPath:(NSString *)keyPath                    ofObject:(id)object                      change:(NSDictionary *)change                     context:(void *)context ;例如:- (void) setAge:(int)theAge{    [self willChangeValueForKey:@"age"];    age = theAge;    [self didChangeValueForKey:@"age"];}+ (BOOL) automaticallyNotifiesObserversForKey:(NSString *)key{    if ([key isEqualToString:@"age"]) {    return NO;    }    return [super automaticallyNotifiesObserversForKey:key];}首先,需要手动实现属性的 setter 方法,并在设置操作的前后分别调用 willChangeValueForKey: 和 didChangeValueForKey方法,这两个方法用于通知系统该 key 的属性值即将和已经变更了;其次,要实现类方法 automaticallyNotifiesObserversForKey,并在其中设置对该 key 不自动发送通知(返回 NO 即可)。这里要注意,对其它非手动实现的 key,要转交给 super 来处理。总结一下,想使用KVO有三种方法:1)使用了KVC使用了KVC,如果有访问器方法,则运行时会在访问器方法中调用will/didChangeValueForKey:方法;没用访问器方法,运行时会在setValue:forKey方法中调用will/didChangeValueForKey:方法。2)有访问器方法运行时会重写访问器方法调用will/didChangeValueForKey:方法。因此,直接调用访问器方法改变属性值时,KVO也能监听到。3)显示调用will/didChangeValueForKey:方法。总之,想使用KVO,只要有will/didChangeValueForKey:方法就可以了。摘自:http://www.mamicode.com/info-detail-515516.html    http://blog.csdn.net/wzzvictory/article/details/9674431
0 0
原创粉丝点击