本文主要是介绍KVO/KVC总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
1.KVC和KVO的概念
1.1 KVC:
NSKeyValueCoding的简称,是一种可以直接通过字符串的名字(key)来访问类属性的机制,而不是通过调用的Setter、Getter方法访问。
1.2 KVO:
NSKeyValueObserving的简称,当指定的对象的属性被修改了,允许对象接收到通知的机制。
2 详述
在很多时候接触到很多地方都有对 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 会提供自动的消息通知。这样开发人员不需要自己去实现这样的方案:每次属性改变了就发送消息通知。
SEL sel = sel_get_uid("setValue:forKey:");
KVC在调用方法setValue的时候
KVO 的优点:
这是 KVO 机制提供的最大的优点。因为这个方案已经被明确定义,获得框架级支持,可以方便地采用。
开发人员不需要添加任何代码,不需要设计自己的观察者模型,直接可以在工程里使用。
其次,KVO 的架构非常的强大,可以很容易的支持多个观察者观察同一个属性,以及相关的值。
KVC 的实现分析
KVC 运用了一个 isa-swizzling 技术。
isa-swizzling 就是类型混合指针机制。KVC 主要通过 isa-swizzling,来实现其内部查找定位的。
isa 指针,就是 is a kind of 的意思,指向维护分发表的对象的类。该分发表实际上包含了指向实现类中的方法的指针和其它数据。
如下 KVC 的代码:
[person setValue:@"personName" forKey:@"name"];
就会被编译器处理成:
IMP method = objc_msg_lookup (person->isa,sel);
method(person,sel,@"personName",@"name");
其中:
SEL数据类型:它是编译器运行Objective-C里的方法的环境参数。
IMP数据类型:他其实就是一个编译器内部实现时候的函数指针。当Objective-C编译器去处理实现一个方法的时候,就会指向一个IMP对象,这个对象是C语言表述的类型。
(1)首先根据方法名找到运行方法的时候所需要的环境参数。
(2)他会从自己isa指针结合环境参数,找到具体的方法实现的接口。
(3)再直接查找得来的具体的方法实现。
这样的话前面介绍的KVO实现就好理解了
当一个对象注册了一个观察者,被观察对象的isa指针被修改的时候,isa指针就会指向一个中间类,而不是真实的类。
所以isa指针其实不需要指向实例对象真实的类。所以我们的程序最好不要依赖于isa指针。在调用类的方法的时候,最好要明确对象实例的类名。
这样只有当我们调用KVC去访问key值的时候KVO才会起作用。所以肯定确定的是,KVO是基于KVC实现的。
下面实例演示一下:
Person.m:
-(void)changeName
{name=@"changeName directly";
}
PersonMonitor.m:
//观察者需要实现的方法
- (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]);}
}
main.m
int main(int argc, const char * argv[])
{@autoreleasepool {//测试代码Person *p =[[Person alloc] init];//监视对象//观察者对象PersonMonitor *pm= [[PersonMonitor alloc]init];//Observer KVO收到通知指定的关键路径相对于接收器。/**pm 对象注册KVO通知。观察者必须实现键值观察方法observeValueForKeyPath。*forKeyPath 关键路径,相对于接收器,用于观察的属性。这个值不能为零。*options NSKeyValueObservingOptions的组合值,指定包含在观察通知的内容。*context 任意的数据传递给anObserver在observeValueForKeyPath。*/[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]; //最后一次修改是直接修改,所以没法产生通知。}return 0;
}
控制台运行结果:
2013-05-23 21:57:01.917 KVC_KVOTest[468:303] p.name is (null)
2013-05-23 21:57:01.919 KVC_KVOTest[468:303] change happen,old:<null> new:name kvc
2013-05-23 21:57:01.920 KVC_KVOTest[468:303] p name get by kvc is name kvc
2013-05-23 21:57:01.920 KVC_KVOTest[468:303] change happen,old:name kvc new:name change by .name=
3 结语
以上是所有内容,希望对大家有所帮助。Demo下载地址:http://download.csdn.net/detail/u010013695/5438389
这篇关于KVO/KVC总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!