KVO和KVC

KVO(NSKeyValueObserving 的简称)、KVC(NSKeyValueCoding 的简称)

在很多时候接触到很多地方都有对 KVC,KVO 的描述,但是都是一笔带过。只知道这是Object-C提供的一个不错的机制,可以很好的减少代码。

首先我们先了解下 KVO 的机制,KVO:当指定的对象的属性被修改了,允许对象接收到通知的机制。每当在类中定义一个监听 

如: [self addObserver:self forKeyPath:@"items" options:0 context:contexStr];

当然你还可以监听其他对象的属性变化,如:[person addObserver:money forKeyPath:@"account" options:0 context:contexStr];

只要当前类中 items 这个属性发生的变化都会触发到以下的方法

- (void)observeValueForKeyPath:(NSString *)keyPath
  ofObject:(id)object
  change:(NSDictionary *)change
  context:(void *)context
 
KVO 的优点:
 
当有属性改变,KVO 会提供自动的消息通知。这样开发人员不需要自己去实现这样的方案:每次属性改变了就发送消息通知。

这是 KVO 机制提供的最大的优点。因为这个方案已经被明确定义,获得框架级支持,可以方便地采用。

开发人员不需要添加任何代码,不需要设计自己的观察者模型,直接可以在工程里使用。

其次,KVO 的架构非常的强大,可以很容易的支持多个观察者观察同一个属性,以及相关的值。

KVC 的实现分析

KVC 运用了一个 isa-swizzling 技术。

isa-swizzling 就是类型混合指针机制。KVC 主要通过 isa-swizzling,来实现其内部查找定位的。

isa 指针,就是 is a kind of 的意思,指向维护分发表的对象的类。该分发表实际上包含了指向实现类中的方法的指针和其它数据。

如下 KVC 的代码:

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

就会被编译器处理成:
  
SEL sel = sel_get_uid("setValue:forKey:");

IMP method = objc_msg_lookup (person->isa,sel);

method(person,sel,@"personName",@"name");

其中:

SEL数据类型:它是编译器运行Objective-C里的方法的环境参数。

IMP数据类型:他其实就是一个编译器内部实现时候的函数指针。当Objective-C编译器去处理实现一个方法的时候,就会指向一个IMP对象,这个对象是C语言表述的类型。
 
KVC在调用方法setValue的时候

(1)首先根据方法名找到运行方法的时候所需要的环境参数。

(2)他会从自己isa指针结合环境参数,找到具体的方法实现的接口。

(3)再直接查找得来的具体的方法实现。

这样的话前面介绍的KVO实现就好理解了

当一个对象注册了一个观察者,被观察对象的isa指针被修改的时候,isa指针就会指向一个中间类,而不是真实的类。

所以isa指针其实不需要指向实例对象真实的类。所以我们的程序最好不要依赖于isa指针。在调用类的方法的时候,最好要明确对象实例的类名。

这样只有当我们调用KVC去访问key值的时候KVO才会起作用。所以肯定确定的是,KVO是基于KVC实现的。

下面实例演示一下:

@implementation Person 

@synthesize name,age;//属性name将被监视

-(void)changeName

   name=@"changeName directly";

@end

@implementation PersonMonitor

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

    if([keyPath isEqual:@"name"])
    { 
        NSLog(@"change happen,old:%@ new:%@",[change objectForKey:NSKeyValueChangeOldKey],[change objectForKey:NSKeyValueChangeNewKey]); 
    } 
}

@end 

//测试代码

Person *p =[[Person alloc] init];//监视对象 

PersonMonitor *pm= [[PersonMonitor alloc]init]; 

[p addObserver:pm forKeyPath:@"name" options:(NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld) context:nil];

//测试前的数据

NSLog(@"p.name is %@",p.name); 

//通过setvalue 的方法,PersonMonitor的监视将被调用 

[p setValue:@"name kvc" forKey:@"name"]; 

//查看设置后的值

NSLog(@"p name get by kvc is %@",[p valueForKey:@"name"]);

//效果和通过setValue 是一致的

p.name=@"name change by .name=";

//通过person自己的函数来更改 name

[p changeName];  

//最后一次修改是直接修改,所以没法产生通知。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Mr__Hacker

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值