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
- (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这个函数。
kvc除了访问私有变量这个用处外,还可以用于字典转模型。在Person
类对外提供一个接口,将转模型的工作放在模型中进行
2,字典转模型
1,字典转模型的时候,字典中的某一个key一定要在模型中有对应的属性
2,如果一个模型中包含了另外的模型对象,是不能直接转化成功的。
3,通过kvc转化模型中的模型,也是不能直接转化成功的。
既然可以通过kvc赋值,同样的也可以通过它进行取值。
NSLog(@"name=%@",[p2 valueForKey:@"name"]);
NSLog(@"dogweight=%@", [p2.dog valueForKeyPath:@"@p2.dog"]);
kvc的查询机制
- 程序优先调用"name"代码来获取该 getter方法 的返回值
- 如果没有getter方法,那么寻找该类名为: _name的成员变量。
- 如果没有_name成员变量,就继续寻找,那么就寻找该类名为 name的成员变量。
- 如果上面的三条都没有达到,那么系统会自动调用该对象的 valueforUndefinedKey:方法。
二,KVO
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:[NSNumber numberWithInt:h+1] 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和NSNotification的区别:
和delegate一样,KVO和NSNotification的作用也是类与类之间的通信,与delegate不同的是1)这两个都是负责发出通知,剩下的事情就不管了,所以没有返回值;2)delegate只是一对一,而这两个可以一对多。这两者也有各自的特点。