KVC和KVO

KVC和KVO都属于键值编程而且底层实现机制都是isa-swizzing

一.KVC概述

1.kvc 是一种通过(key)来访问类属性的机制,而不是通过 set get 方法。

2.关键方法定义在NskeyValueCodingProtocol

kvc 支持类对象,和基本数据类型

二。

KVC使用

举一个例子:

我创建一个类:

@interface Pperson :NSObject

@end

@implementation Pperson

-(NSString*)description{

return[NSStringstringWithFormat:@"PPdescrpation<%@:%p>,{name:%@,age:%d}",[selfclass],self,self.name,self.age];

}

@end

@implementationViewController

-(void)aboutPperson{

Pperson*p1 = [[Ppersonalloc]init];

[p1setValue:@"junjun"forKey:@"name"];

[p1setValue:@"26"forKey:@"age"];

Pperson*p2 = [[Ppersonalloc]init];

[p2setValue:@"junjun1"forKey:@"name"];

[p2setValue:@"27"forKey:@"age"];

NSLog(@"p1=====%@,p2.name====%@",p1,[p2valueForKeyPath:@"name"]);

NSArray*arr =@[p1,p2];

///可以通过路径直接找到属性的值无论怎么嵌套属性

NSLog(@"arr的name===%@",[arrvalueForKeyPath:@"name"]);

}

运行结果:p1=====PPdescrpation,{name:junjun,age:26},p2.name====junjun1

2017-02-17 10:57:46.471 NewCocoTest[1063:47085] arr的name===(

junjun,

junjun1

)

我把Pperson两个属性改为在.m里面

@interfacePperson(){

NSString*name;

int age;

}

@end

@implementationPperson

-(NSString*)description{

return[NSStringstringWithFormat:@"PPdescrpation<%@:%p>,{name:%@,age:%d}",[selfclass],self,name,age];

}

//运行结果如下:

p1=====PPdescrpation,{name:junjun,age:26},p2.name====junjun1

2017-02-17 10:47:44.601 NewCocoTest[1012:40918] arr的name===(

junjun,

junjun1

)

两次的运行结果是一样的,所以说 KVC 不是调用的set get 方法,还有KVC 可以层层深入直接找到属性的值。把代码,[arrvalueForKeyPath:@"name"]换成[arrvalueForKey:@"name"]也是可以的

再创建一个类作为Pperson类的变量

@interfaceCourseModel(){

NSString*_coursename;

}

@end

@implementationCourseModel

-(NSString*)description{

return[NSStringstringWithFormat:@"coursename=====%@",_coursename];

}

-(void)aboutPperson{

Pperson*p1 = [[Ppersonalloc]init];

CourseModel*course1 = [[CourseModelalloc]init];

[course1setValue:@"语文课"forKey:@"coursename"];

[p1setValue:@"junjun"forKey:@"name"];

[p1setValue:@"26"forKey:@"age"];

[p1setValue:course1forKey:@"course"];

//NSString *name1 = [p1 valueForKeyPath:@"course.coursename"];

Pperson*p2 = [[Ppersonalloc]init];

[p2setValue:@"junjun1"forKey:@"name"];

[p2setValue:@"27"forKey:@"age"];

CourseModel*course2 = [[CourseModelalloc]init];

[course2setValue:@"数学课"forKey:@"coursename"];

[p2setValue:course2forKey:@"course"];

NSLog(@"p1=====%@,p2.name====%@",p1,p2);

NSArray*arr =@[p1,p2];

NSArray*namearr = [[arrvalueForKeyPath:@"course"]valueForKeyPath:@"coursename"];

for(inti =0; i < namearr.count; i++) {

UILabel*label = [[UILabelalloc]init];

label.text= namearr[i];

label.frame=CGRectMake(i*100,100,100,30);

label.backgroundColor= [UIColorredColor];

[self.viewaddSubview:label];

}

}

运行结果如下:

p1=====PPdescrpation,{name:junjun,age:26,course:coursename=====语文课},p2.name====PPdescrpation,{name:junjun1,age:27,course:coursename=====数学课}

@end

因此证明了一个道理:KVC可以一层一层的深入寻找属性的值。我们用NSString*类型设置的属性值@"26",而我们的属性是NSInteger类型的,存取都没有问题。

我又加了一个数组在PPerson类里面

@interfacePperson(){

NSString*name;

intage;

CourseModel*course;

NSArray*otherStudent;//其他学生

}

-(void)aboutPperson{

Pperson*p = [[Ppersonalloc]init];

[psetValue:@"45"forKey:@"age"];

Pperson*p1 = [[Ppersonalloc]init];

CourseModel*course1 = [[CourseModelalloc]init];

[course1setValue:@"语文课"forKey:@"coursename"];

[p1setValue:@"junjun"forKey:@"name"];

[p1setValue:@"26"forKey:@"age"];

[p1setValue:course1forKey:@"course"];

//NSString *name1 = [p1 valueForKeyPath:@"course.coursename"];

Pperson*p2 = [[Ppersonalloc]init];

[p2setValue:@"junjun1"forKey:@"name"];

[p2setValue:@"27"forKey:@"age"];

CourseModel*course2 = [[CourseModelalloc]init];

[course2setValue:@"数学课"forKey:@"coursename"];

[p2setValue:course2forKey:@"course"];

NSLog(@"p1=====%@,p2.name====%@",p1,p2);

NSArray*arr =@[p1,p2];

NSArray*namearr = [arrvalueForKeyPath:@"course.coursename"];

//[[arrvalueForKeyPath:@"course"] valueForKeyPath:@"coursename"];//或者

for(inti =0; i < namearr.count; i++) {

UILabel*label = [[UILabelalloc]init];

label.text= namearr[i];

label.frame=CGRectMake(i*100,100,100,30);

label.backgroundColor= [UIColorredColor];

[self.viewaddSubview:label];

}

NSArray*arryother = [[NSArrayalloc]initWithObjects:p1,p2,nil];

[psetValue:arryotherforKey:@"otherStudent"];

NSLog(@"其他学生的分数%@",[pvalueForKeyPath:@"otherStudent.age"]);//valueForKeyPath:@"age"]);

NSLog(@"最大年龄%@",[pvalueForKeyPath:@"[email protected]"] );

NSLog(@"最小年龄%@",[pvalueForKeyPath:@"[email protected]"] );

NSLog(@"年龄的平均值%@",[pvalueForKeyPath:@"[email protected]"]);

}

//运行结果如下:NewCocoTest[1992:130252]其他学生的分数(

26,

27

)

2017-02-17 15:01:54.655 NewCocoTest[1992:130252]最大年龄27

2017-02-17 15:01:54.655 NewCocoTest[1992:130252]最小年龄26

2017-02-17 15:01:54.655 NewCocoTest[1992:130252]年龄的平均值26.5

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

KVO的使用也很简单,就是简单的3步。

1.注册需要观察的对象的属性addObserver:forKeyPath:options:context:

2.实现observeValueForKeyPath:ofObject:change:context:方法,这个方法当观察的属性变化时会自动调用

3.取消注册观察removeObserver:forKeyPath:context:

举一个简单的例子:

@interfacetestMyPerson :NSObject{

NSString*testMyPersonName;

}

@end

在控制器里调用这个方法

-(void)testKVO{

self.testPeron= [[testMyPersonalloc]init];

[self.testPeronaddObserver:selfforKeyPath:@"testMyPersonName"options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOldcontext:nil];

}

-(void)observeValueForKeyPath:(NSString*)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void*)context{

if([keyPathisEqualToString:@"testMyPersonName"]) {

NSLog(@"new=====%@",[changevalueForKey:NSKeyValueChangeNewKey]);

}else{

}

}

-(void)Uiconfig{

self.title=@"第二个控制器";

UIButton*button = [[UIButtonalloc]init];

[self.viewaddSubview:button];

button.frame=CGRectMake(0,64,100,30);

[buttonsetTitle:@"高度变化"forState:UIControlStateNormal];

[buttonsetTitleColor:[UIColorredColor]forState:UIControlStateNormal];

[buttonaddTarget:selfaction:@selector(changeHeight:)forControlEvents:UIControlEventTouchUpInside];

}

-(void)changeHeight:(UIButton*)button{

NSString*height = [self.testPeronvalueForKey:@"testMyPersonName"] ;

[self.testPeronsetValue:@"我是俊俊"forKey:@"testMyPersonName"];

NSLog(@"testMyPersonNameName == ===%@",[self.testPeronvalueForKeyPath:@"testMyPersonName"]);

}


当属性值无论会不会改变都会调用一遍监听方法,

NewCocoTest[3097:194495] new=====我是俊俊

2017-02-17 17:08:15.560 NewCocoTest[3097:194495] testMyPersonNameName == ===我是俊俊

2017-02-17 17:08:22.140 NewCocoTest[3097:194495] new=====我是俊俊

2017-02-17 17:08:34.438 NewCocoTest[3097:194495] testMyPersonNameName == ===我是俊俊

你可能感兴趣的:(KVC和KVO)