以实例讲解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)