以实例讲解Objective-C中的KVO与KVC机制
KVO实例浅析
最近遇到个问题,在处理项目中一个评论界面时,因为直接用的是UIWebView展示评论列表,结果取到的页面上下都有一段CGSize为(320,65)的乱七八糟的广告,十分碍眼.头部广告因很方便的在头部坐标贴上自己的logo解决了,但是尾部的,因为每个页面的评论长短不一,坐标也就不一样,这样就不能给定死坐标去贴logo,思前想后,通过KVO很好的解决了这个问题.
@KVO概述:
KVO,即:Key-ValueObserving,它提供一种机制,当指定的对象的属性被修改后,则对象就会接受到通知。
简单的说就是每次指定的被观察的对象的属性被修改后,KVO就会自动通知相应的观察者了。
使用步骤如下:
1.注册,指定被观察者的属性,
2.实现回调方法
3.触发回调方法
4.移除观察
代码实例:
-(void)viewDidLoad{
//KVO,作为一个观察者,只要属性"contentSize"发生变化,回调方法里面就会通知
[_webView.scrollViewaddObserver:selfforKeyPath:@"contentSize"options:NSKeyValueObservingOptionNewcontext:NULL];
}
// 回调方法
-(void)observeValueForKeyPath:(NSString*)keyPathofObject:(id)objectchange:(NSDictionary*)changecontext:(voidvoid*)context
{
if(object==_webView.scrollView&&[keyPathisEqualToString:@"contentSize"])
{
// 得到最大的Y坐标
CGSizesize=_webView.scrollView.contentSize;
if(size.height>568.0){
//遮挡广告
_hideBottomImage=[[UIImageViewalloc]initWithFrame:CGRectMake(0,size.height-67,ScreenWidth,67)];
_hideBottomImage.image=[UIImageimageNamed:@"banner"];
[_webView.scrollViewaddSubview:_hideBottomImage];
[_hideBottomImagerelease];
}
}
else
{
// 调用父类的方法
[superobserveValueForKeyPath:keyPathofObject:objectchange:changecontext:context];
}
}
-(void)dealloc{//---->在ARC环境下也能调用dealloc方法,只是不需要写[superdealloc]
//移除KVO,否则会引起资源泄露
[_webView.scrollViewremoveObserver:selfforKeyPath:@"contentSize"];
[superdealloc];
} 上面是针对contentSize属性,其他属性依此类推
KVC
通常,我们都是通过属性的set和get方法来赋值和取值,这里介绍用Key-Value-Coding(KVC)键值编码来给类的属性赋值和取值.
1.基本方式(setValue:forKey: valueForKey)
// ---定义一个Student类(.m文件无任何操作)
#import<Foundation/Foundation.h>
@classHMTClass;
@interfaceHMTStudent:NSObject{
NSString*_name;
BOOL_test;
BOOL_isTest;
BOOLtest;
BOOLisTest;
}
@property(nonatomic,copy)NSString*name;
@property(nonatomic,copy)NSString*sex;
@property(nonatomic,assign)NSIntegerage;
@property(nonatomic,strong)HMTClass*hmtClass;
@end
// ---main文件
HMTStudent*student=[[HMTStudentalloc]init];
student.hmtClass=[[HMTClassalloc]init];
student.name=@"humingtao”; // set方法赋值
// KVC赋值
[studentsetValue:@“maweiisdog"forKey:@"name”];
[studentsetValue:@"m"forKey:@"sex"];
[studentsetValue:@(10)forKey:@"age"];
// 取值
NSLog(@"%s__%d__|%@",__FUNCTION__,__LINE__,[studentvalueForKey:@"name"]);
特别注意:
我在类里面还定义了4个BOOL值变量,用来验证KVC访问属性键顺序
[studentsetValue:@(YES)forKey:@"test”];
结果是:_test—>_isTest—>test—>isTest
2.键路径访问(用于一个类中属性的属性setValue:ForKeyPath:forKeyPath)
// 创建一个班级类 @interfaceHMTClass:NSObject @property(nonatomic,copy)NSString*name; @end
然后前面第一点中在Student类中写了一个班级属性hmtClass
HMTClass*hmtClass=[[HMTClassalloc]init]; [hmtClasssetValue:@"宇宙一班"forKey:@"name"]; [studentsetValue:hmtClassforKey:@"hmtClass"]; NSString*hmtClassName=[studentvalueForKeyPath:@"hmtClass.name"]; //也可以这样存值 [studentsetValue:@"宇宙一班"forKeyPath:@"hmtClass.name"]; student.hmtClass.name=[studentvalueForKeyPath:@"hmtClass.name"];
3.自动封装基本数据类型
我们在Student类中添加分数属性NSIntegernumber学号;
#import<Foundation/Foundation.h>
@classHMTClass;
@interfaceHMTStudent:NSObject
{
NSString*_name;
NSIntegernumber;
}
@end
[studentsetValue:@"100"forKeyPath:@"number"];
NSString*number=[studentvalueForKey:@"number"]; 可见用NSString*类型设置的属性值@"100",而我们的属性是NSInteger类型的,存取都没有问题。
4.操作集合
在Student类中加入数组NSArray,用来表示其他的学生。
#import<Foundation/Foundation.h>
@classHMTClass;
@interfaceHMTStudent:NSObject
{
NSArray*manyStudents;
}
@end
Student*student1=[[HMTStudentalloc]init];
Student*student2=[[HMTStudentalloc]init];
Student*student3=[[HMTStudentalloc]init];
[student1setValue:@"200"forKey:@"number"];
[student2setValue:@"300"forKey:@"number"];
[student3setValue:@"400"forKey:@"number"];
NSArray*array=[NSArrayarrayWithObjects:student1,student2,student3,nil];
[studentsetValue:arrayforKey:@"manyStudents"];
NSLog(@"%@",[studentvalueForKeyPath:@"manyStudents.number"]); 打印出来是数组(200,300,400)