来源:  对月流

链接:http://www.jianshu.com/p/f1393d10109d

写在前面:

关于KVC和KVO各种博客多了去了,重新整理下,就当是温习一下吧,也还算是个新手,不对的地方请指教,喜欢的点个喜欢什么也是挺好。

一,KVC

KVC也就是key-value-coding,即键值编码,通常是用来给某一个对象的属性进行赋值,例如有人这么一个类,其对外有两个属性,姓名和年龄,我们在创建了一个人p后可以通过点语法直接给p赋值。

  1. Person *p = [[Person alloc] init];
  2. p.name = @"张三";
  3. p.age = ;

我们也可以通过kvc给这个人p赋值,代码如下,因为setValue这里的值是id类型的,所以将整数包装成一个对象

  1. [p setValue:@"张三" forKey:@"name"];
  2. [p setValue:@ forKey:@"age"];

但是我们这样去赋值显得多此一举,可是如果人这个类的属性是没有暴露在外面呢?比如现在给人这个类一个私有的身高的属性,并且对外提供一个输出身高的接口,如下

  1. #import "Person.h"
  2. @implementation Person
  3. {
  4. NSInteger _height;
  5. }
  6.  
  7. - (void)logHeight
  8. {
  9. NSLog(@"%ld",_height);
  10. }
  11. @end

这时候我们是没有办法去给人p直接设置身高的,外面我们访问不到它.但是有了kvc就不一样了。

  1. [p setValue:@ forKey:@"height"];

我们通过kvc可以直接对私有属性进行赋值,打印如下

除了[p setValue:@170 forKey:@"height"]这个方法外,还有一个方法也是可以对私有属性进行赋值的[p setValue:@170 forKeyPath:@"height"];这两个方法对于一个普通的属性是没有区别的,都可以用,但是对于一些特殊的属性就有区别了。

比如说人这个类有个属性是狗,狗又有属性体重。

  1. p.dog = [[Dog alloc] init];
  2. [p setValue:@ forKey:@"dog.weight"];

如果我们直接这样是会报错说找不到dog.weight这个key的,而在storyboard中,我们拖控件连线错误的时候也会报错说找不到什么key,说明storyboard在赋值的时候也是通过kvc来操作的。

这里如果我们换另外的一个方法,这时候是不会报错的,而且可以打印出狗的体重.

  1. [p setValue:@200forKeyPath:@"dog.weight"];

说明forKeyPath是包含了forKey这个方法的功能的,甚至forKeyPath方法还有它自己的高级的功能,它会先去找有没有dog这个key,然后去找有没有weight这个属性。所以我们在使用kvc的时候,最好用forKeyPath这个方法。

最后还有一点,如下代码

  1. [p setValue:@ forKey:@"height"];

我们传入的字符串key是height,但是定义的属性是_height,但是通过kvc还是可以给_height属性赋到值。说明对某一个属性进行赋值,可以不用加下划线,而且它的查找规则应该是:先查找和直接写入的字符串相同的成员变量,如果找不到就找以下划线开头的成员变量。

kvc除了访问私有变量这个用处外,还可以用于字典转模型。在Person类对外提供一个接口,将转模型的工作放在模型中进行

  1. - (instancetype)initWithDict:(NSDictionary *)dict
  2. {
  3.  
  4. if (self = [super init]) {
  5.  
  6. [self setValuesForKeysWithDictionary:dict];
  7.  
  8. }
  9. returnself;
  10. }

外面可以直接将字典传入,和平常转模型相比,kvc更加方便,减少了代码量。

  1. NSDictionary*PersonDict = @{@"name":@"李四",@"age":@""};
  2. Person *p2 = [Person personWithDict:PersonDict];
  3. NSLog(@"name = %@,age =%ld",p2.name,p2.age);

所以kvc最常见的两种用法就是:

1,对私有变量进行赋值

2,字典转模型

但是也有一些需要注意的地方

1,字典转模型的时候,字典中的某一个key一定要在模型中有对应的属性

2,如果一个模型中包含了另外的模型对象,是不能直接转化成功的。

3,通过kvc转化模型中的模型,也是不能直接转化成功的。

既然可以通过kvc赋值,同样的也可以通过它进行取值。

  1. NSLog(@"name=%@",[p2valueForKey:@"name"]);
  2. NSLog(@"dogweight=%@", [p2.dogvalueForKeyPath:@"@p2.dog"]);

二,KVO

KVO,即key-value-observing,利用一个key来找到某个属性并监听其值得改变。其实这也是一种典型的观察者模式。

简单的说,kvo的用法非常简单。

1,添加观察者

2,在观察者中实现监听方法,observeValueForKeyPath: ofObject: change: context:(通过查阅文档可以知道,绝大多数对象都有这个方法,因为这个方法属于NSObject)

3,移除观察者

具体代码如下:

  1. //让对象b监听对象a的name属性
  2. //options属性可以选择是哪个
  3. /*
  4. NSKeyValueObservingOptionNew =
  5. 0x01, 新值
  6. NSKeyValueObservingOptionOld =
  7. 0x02, 旧值
  8. */
  9. [a addObserver:b forKeyPath:@"name"options:kNilOptionscontext:nil];
  10. a.name = @"zzz";
  11.  
  12. #pragma mark - 实现KVO回调方法
  13. /*
  14. * 当对象的属性发生改变会调用该方法
  15. * @param keyPath 监听的属性
  16. * @param object 监听的对象
  17. * @param change 新值和旧值
  18. * @param context 额外的数据
  19. */
  20.  
  21. - (void)observeValueForKeyPath:(NSString *)keyPath
  22. ofObject:(id)object
  23. change:(NSDictionary
  24. *)change context:(void *)context
  25. {
  26. NSLog(@"%@的值改变了,",keyPath);
  27. NSLog(@"change:%@", change);
  28. }
  29.  
  30. //最后不要忘记,和通知一样,要在dealloc方法里面移除监听
  31.  
  32. - (void)dealloc
  33. {
  34.  
  35. [a removeObserver:bforKeyPath:@"name"];
  36.  
  37. }

KVO的底层实现

当一个类的属性被观察的时候,系统会通过runtime动态的创建一个该类的派生类,并且会在这个类中重写基类被观察的属性的setter方法,而且系统将这个类的isa指针指向了派生类,从而实现了给监听的属性赋值时调用的是派生类的setter方法。重写的setter方法会在调用原setter方法前后,通知观察对象值得改变。

具体实现图如下,这里我拿的是iOS程序猿的图,借用一下应该没关系吧?

最后

貌似有个facebook开源的工具,KVOController ,是一个简单安全的 KVO(Key-value Observing,键-值 观察)工具,好像挺好用的。

iOS:KVC和KVO的更多相关文章

  1. iOS KVC 和 KVO 的学习

    KVC  (NSKey Value Coding) :键值编码 KVO (Key Value Observing) :键值监听 前言:我曾经用过 监听 一个音频何时结束 监听视频播放 状态等 用了这种 ...

  2. [iOS] KVC 和 KVO

    开发iOS经常会看见KVO和KVC这两个概念,特地了解了一下. 我的新博客wossoneri.com link KVC Key Value Coding KVC是一种用间接方式访问类的属性的机制.比如 ...

  3. iOS KVC 和 KVO 区别简单总结

    KVC: key value coding,键值编码.是一种通过使用属性的名称(key)来间接访问对象属性的方法.这个方法可以不用通过 setter/getter 方法来访问对象的属性.该方法使用的实 ...

  4. 【原】iOS中KVC和KVO的区别

    在iOS开发中经常会看到KVC和KVO这两个概念,比较可能混淆,特地区分一下 KVC(Key Value Coding) 1> 概述 KVC:Key Value Coding,键值编码,是一种间 ...

  5. iOS中关于KVC与KVO知识点

    iOS中关于KVC与KVO知识点 iOS中关于KVC与KVO知识点  一.简介 KVC/KVO是观察者模式的一种实现,在Cocoa中是以被万物之源NSObject类实现的NSKeyValueCodin ...

  6. iOS开发中KVC、KVO简介

    在iOS开发中,KVC和KVO是经常被用到的.可以使用KVC对对象的属性赋值和取得对象的属性值,可以使用KVO监听对象属性值的变化.简单介绍一下KVC和KVO. 一:键值编码(KVC) KVC,全称 ...

  7. iOS开发-KVC和KVO的理解

    KVC和KVO看起来很专业,其实用起来还是比较简单的,KVC(Key-value coding)可以理解为键值对编码,如果对象的基本类型,那么键值对编码实际上和get,set方法没有区别,如果是属性是 ...

  8. iOS编程——Objective-C KVO/KVC机制[转]

    这两天在看和这个相关的的内容,全部推翻重写一个版本,这是公司内做技术分享的文档总结,对结构.条理做了更清晰的调整.先找了段代码,理解下,网上看到最多的一段的关于KVC的代码 先上代码 1.     1 ...

  9. 【iOS】KVC 与 KVO

    一.KVC与KVO *"KVC":key value Coding(键值编码) *目的:间接的改动或获取对象的属性,减少程序(类与类)之间的耦合度. *"KVO" ...

随机推荐

  1. 解决Myeclipse编译不生成.class文件问题

    1.Project --> clean...  如果该操作无效,请执行2. 2.Preferences -->Java -->Compliler -->Building --& ...

  2. linux命令(29):cd命令

    例1:进入系统根目录 cd  / cd ../.. // [直接退到当前根目录] 例2:使用 cd 命令进入当前用户主目录 cd 例3:跳转到指定目录 cd  /home/test 例4:返回进入此目 ...

  3. go run/ go install/ go build / go get的区别

    go run 运行当个.go文件 go install 在编译源代码之后还安装到指定的目录 go build 加上可编译的go源文件可以得到一个可执行文件 go get = git clone + g ...

  4. Python 分页功能

    自定义分页组件 """ 自定义分页组件的使用方法: pager_obj = Pagination(request.GET.get('page',1),len(HOST_L ...

  5. Combination Sum I&&II(经典的回溯算法题)

    I: Given a set of candidate numbers (C) and a target number (T), find all unique combinations in C w ...

  6. 通过url判断当前页,动态给导航加样式

    //通过url判断当前页,动态给导航加样式 var str =location.pathname; var index = str.lastIndexOf("\/"); str = ...

  7. 如何使用 Java 对 List 中每个对象元素按时间顺序进行排序

    如何使用 Java 对 List 中每个对象元素按时间顺序进行排序 Java 实现 import java.text.SimpleDateFormat; import java.util.ArrayL ...

  8. Intellij IDEA 去掉Mapper文件中的背景

    1.在setting中输入:inspection --> SQL 2.去掉背景颜色,Apply即可

  9. python中,将字符串由utf8转gbk

    uni_str = utf8_str.decode('utf-8'); gbk_str = uni_str.encode('gbk');

  10. java之异常

    package com.text.exception; class Test{ void add(int a,int b) throws Exception { int c; c=a/b; Syste ...