OC动态特性之 — KVC、KVO

由于Objective-C是基于Smalltalk进行设计的,所以它具有动态加载、动态绑定等特性。Key-value coding (KVC) 和 key-value observing (KVO) 是两种能让我们驾驭 Objective-C 动态特性并简化代码的机制。

1.KVC

在ObjC的编程中,我们习惯于通过属性的set和get方法来对属性的值进行读写,其实由于ObjC的语言特性,你根本不必进行任何操作就可以进行属性的动态读写,这种方式就是Key Value Coding(简称KVC)。

KVC的操作方法由NSKeyValueCoding协议提供,而NSObject就实现了这个协议,也就是说ObjC中几乎所有的对象都支持KVC操作,常用的KVC操作方法如下:

写方法:setValue:属性值 forKey:属性名(用于简单路径)、setValue:属性值 forKeyPath:属性路径(用于复合路径,例如Person有一个Account类型的属性,那么person.account就是一个复合属性)

读方法:valueForKey:属性名(简单路径)、valueForKeyPath:属性名(复合路径)

示例代码:

Book.h

#import

@interfaceBook:NSObject{

// price属性

double_price;

}

@end

Book.m

#import "Book.h"

@implementationBook

@end

Person.h

#import

// 声明Book类

@classBook;

#pragma属性:无set和get方法

@interfacePerson:NSObject{

int_age;

Book*_book;

}

#pragma属性:有set和get方法

@property(nonatomic,copy)NSString*name;

@property(nonatomic,assign)floatheight;

@end

Person.m

#import "Person.h"

@implementationPerson

@end

main.m

#import

#import "Person.h"

#import "Book.h"

intmain(intargc,constchar*argv[]){

@autoreleasepool{

Person*person=[[Personalloc]init];

Book*book=[[Bookalloc]init];

// setValue:forKey: 方法设置简单属性的值,value值必须是OC对象

[person setValue:@18forKey:@"_age"];

[person setValue:@1.7forKey:@"height"];

[person setValue:@"jack"forKey:@"name"];

[person setValue:book forKey:@"_book"];

// setValue:forKeyPath: 方法设置复合属性的值

[person setValue:@25.8forKeyPath:@"book.price"];

// valueForKey: 方法获取简单属性的值

intage=[[person valueForKey:@"age"]intValue];

floatheight=person.height;

NSString*name=[person valueForKey:@"_name"];

// valueForKeyPath: 方法获取复合属性的值

doublebookPrice=[[person valueForKeyPath:@"_book._price"]doubleValue];

NSLog(@"age = %d",age);

NSLog(@"height = %f",height);

NSLog(@"name = %@",name);

NSLog(@"bookPrice = %f",bookPrice);

}

return0;

}

小结:

如果是动态设置属性,以上文 age 属性为例,会优先考虑调用 setAge: 方法,如果没有该方法则优先考虑搜索成员变量 _age,如果仍然不存在则搜索成员变量 age,如果最后仍然没搜索到则会调用这个类的setValue:forUndefinedKey:方法(注意搜索过程中不管这些方法、成员变量是私有的还是公共的都能正确设置),所以key值加不加下划线都是可以的。

如果是动态读取属性,则优先考虑调用 age 方法(属性age的getter方法),如果没有搜索到则会优先搜索成员变量 _age,如果仍然不存在则搜索成员变量 age,如果最后仍然没搜索到则会调用这个类的valueforUndefinedKey:方法(注意搜索过程中不管这些方法、成员变量是私有的还是公共的都能正确读取),所以key值加不加下划线都是可以的。

2.KVO

在如今比较流行的MVVM设计模式中,需要有一种双向绑定的机制,在数据模型发生了修改之后立即将改变呈现到UI视图上去。OC中原生的就支持这么一种机制,那就是Key Value Observing(简称KVO)。KVO其实是一种观察者模式,利用它可以很容易实现视图组件和数据模型的分离,当数据模型的属性值改变之后作为监听器的视图组件就会被激发,激发时就会回调监听器自身。在ObjC中要实现KVO则必须实现NSKeyValueObServing协议,不过幸运的是NSObject已经实现了该协议,因此几乎所有的ObjC对象都可以使用KVO。

在ObjC中使用KVO操作常用的方法如下:

注册指定Key路径的监听器:addObserver: forKeyPath: options:  context:

删除指定Key路径的监听器:removeObserver: forKeyPath、removeObserver: forKeyPath: context:

回调监听:observeValueForKeyPath: ofObject: change: context:

KVO的使用步骤也比较简单:

通过addObserver: forKeyPath: options: context:为被监听对象(它通常是数据模型)注册监听器

重写监听器的observeValueForKeyPath: ofObject: change: context:方法

这里我们还是在上面的例子基础上继续扩展,我们为 person 对象添加一个监听者 observer。当我们的 person 对象的 height 属性值变动之后我们希望 observer 可以及时获得通知。

为了认识KVO能监听对象属性值的哪几种方式的变化,我们自己实现一个方法来改变本身属性的值,那么在Person类中做一些改变,添加一个changValue方法:

-(void)changeValue{

_height+=0.1;

NSLog(@"----- 自己实现的方法改变height的值 ------");

NSLog(@"height = %f",_height);

}

创建一个Observer类:

Observer.h

#import

@interfaceObserver:NSObject

@property(nonatomic,copy)NSString*name;

@end

Observer.m

#import "Observer.h"

@implementationObserver

// 这个方法在对象的监听属性发生改变时,会自动调用。监听者对属性发生的改变做出什么反应也体现在这里。

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

NSLog(@"------ %@ 在监听 ------",self.name);

NSLog(@"keyPath: %@",keyPath);

NSLog(@"object: %@",[objectvalueForKey:@"name"]);

NSLog(@"change: %@",change);

NSLog(@"context: %@",context);

}

@end

main.m

#import

#import "Person.h"

#import "Observer.h"

intmain(intargc,constchar*argv[]){

@autoreleasepool{

// 创建一个person对象,设置两个属性值

Person*person=[[Personalloc]init];

person.height=1.7;

person.name=@"jack";

// 创建一个observer对象,设置一个name属性的值

Observer*observer=[[Observeralloc]init];

observer.name=@"observer";

// 为person对象注册一个监听者

/**

*  第1个参数:谁来监听

*  第2个参数:监听哪一个属性

*  第3个参数:属性发生了什么变化

*  第4个参数:额外传入的参数

*/

[person addObserver:observer forKeyPath:@"height"options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOldcontext:@"这里可以传入一些东西..."];

// 通过setter方法改变被监听的属性的值

person.height=1.8;

// 通过KVC方法改变被监听的属性的值

[person setValue:@1.9forKey:@"height"];

// 通过自己实现的changeValue方法改变被监听的属性的值

[person changeValue];

// 移除监听(在新版本编译器中,必须配对调用监听和移除监听的方法,否则程序会崩溃)

/**

*  第1个参数: 要移除哪个监听者

*  第2个参数: 监听的是哪个属性

*/

[person removeObserver:observer forKeyPath:@"height"];

}

return0;

}

从上面的运行结果来看,只有通过setter或KVC修改的属性值,才会调用observeValueForKeyPath:方法,通过其他方式修改属性值并不能通知监听者,这里需要注意。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 20- 枚举,枚举原始值,枚举相关值,switch提取枚举关联值 Swift枚举: Swift中的枚举比OC中的枚...
    iOS_恒仔阅读 6,870评论 1 6
  • KCV 其实由于ObjC的语言特性,你根部不必进行任何操作就可以进行属性的动态读写,这种方式就是Key Value...
    TYM阅读 4,680评论 0 4
  • 目录:1.KVC用法;2.KVC和对象的setter、getter方法的区别;3.key和keyPath的区别;4...
    伦伦子_f7b3阅读 3,728评论 0 1
  • Swift 介绍 简介 Swift 语言由苹果公司在 2014 年推出,用来撰写 OS X 和 iOS 应用程序 ...
    大L君阅读 8,557评论 3 25
  • 如果一个人正在用心去做一件事情,但是他不知道结果会是怎么样,其实他的内心是非常纠结,同时也是非常期待的。当然啦,如...
    未自己阅读 1,040评论 0 0