(十二)KVC/KVO

前言

刚写oc,还不知道kvc,只知道这个东西和前端中的json很像,后来发现确实很像哈哈,不错还有一些前端json不具备的东西。
文章内容整理自KVC 与 KVO 理解,一篇2012年的文章,但是内容写的真的好,心怀敬畏。

一、KVC

KVC,即是指 NSKeyValueCoding,一个非正式的 Protocol,提供一种机制来间接访问对象的属性。KVO 就是基于 KVC 实现的关键技术之一。
一个对象拥有某些属性。比如说,一个 Person 对象有一个 name 和一个 address 属性。以 KVC 说法,Person 对象分别有一个 value 对应他的 name 和 address 的 key。 key 只是一个字符串,它对应的值可以是任意类型的对象。从最基础的层次上看,KVC 有两个方法:一个是设置 key 的值,另一个是获取 key 的值。如下面的例子:

//方法实现
- (void)changName:(card *)card newName:(NSString *)newName
{
    NSString *oldName = [card valueForKey:@"name"];
    [card setValue:newName forKey:@"name"];
    NSString *Name = [card valueForKey:@"name"];
    NSLog(@"change name from %@ to %@",oldName,Name);
}
//方法调用
card *BMW = [card new];
BMW.name = @"宝马";
[self changName:BMW newName:@"奔驰"];
//调用结果
textview[1668:192219] change name from 宝马 to 奔驰

以上代码采用valueForKey和setValue forKey 分别来获取value和改变value,讲card的name从宝马改到奔驰。

key和keyPat的区别
抱歉我搞错了card和car,本来是用car,毕竟例子都用了宝马和奔驰
如果card中有个key本身是一个对象,对象还有一个属性,像这样

@interface card : NSObject

@property (nonatomic,strong) NSString *name;
@property (nonatomic,strong) Engine *Engine;

@end

@interface Engine : NSObject

@property (nonatomic,strong) NSString *EngineName;

@end

那么我们可以这样使用

- (void)logKey:(card *)card
{
    NSString *name = [card valueForKey:@"name"];
    //NSString *Engine = [[card valueForKey:@"Engine"] valueForKey:@"EngineName"];
    NSString *Engine = [card valueForKeyPath:@"Engine.EngineName"];
    NSLog(@"name = %@,Engine = %@",name,Engine);
}

上段代码中注释掉的那行和下面一样代码作用相同。
打印出来是这样的

textview[1890:211955] name = 奔驰,Engine = V10

二、KVO

我们想来创建一个cardWatch作为card的监听,只实现一个小小小的功能,当name改变时打印出来(模仿原博客,但是很多地方我实现的不好就改的简单了。)

//定义,将注册监听暴露出来
#import <Foundation/Foundation.h>
#import "card.h"

@interface cardWatch : NSObject

-(void) watchPersonForChangeOfAddress:(card *)p;

@end
//实现,数组这里的作用我猜作者是想保留到注册的监听的对象,可以实现注销
@interface cardWatch ()
{
    NSMutableArray *m_observedPeople;
}

@end

@implementation cardWatch

-(void) watchPersonForChangeOfAddress:(card *)p
{
    
    // this begins the observing
    [p addObserver:self
        forKeyPath:@"name"
           options:0
           context:nil];
    
    // keep a record of all the people being observed,
    // because we need to stop observing them in dealloc
    [m_observedPeople addObject:p];
}

// whenever an observed key path changes, this method will be called
- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary *)change
                       context:(void *)context

{
    // use the context to make sure this is a change in the address,
    // because we may also be observing other things
    NSLog(@"have a new name %@",[object valueForKey:@"name"]);
}

-(void) dealloc;
{
    
    // must stop observing everything before this object is
    // deallocated, otherwise it will cause crashes
    for(card *p in m_observedPeople){
        [p removeObserver:self forKeyPath:@"name"];
    }
    
    [m_observedPeople removeAllObjects];
    m_observedPeople = nil;
    
   // [super dealloc];
    
}

-(id) init;
{
    if(self = [super init]){
        m_observedPeople = [NSMutableArray new];
    }
    
    return self;
}
//这就是调用了啦
BMW.name = @"宝马";
cardWatch *watch = [cardWatch new];
[watch watchPersonForChangeOfAddress:BMW];
BMW.name = @"飞度";
BMW.name = @"思域";
BMW.name = @"手扶拖拉机";
//打印如我们想的一样,开心
2017-07-03 17:34:28.204 textview[2079:246465] have a new name 飞度
2017-07-03 17:34:28.204 textview[2079:246465] have a new name 思域
2017-07-03 17:34:28.205 textview[2079:246465] have a new name 手扶拖拉机

恩,基本上就是这样了,这个监听感觉和ios的广播有点像,这也是前端json不具备的,感觉KVO给了这个数据结构生命,很厉害。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容