KVC之-setValue:forKey:方法实现原理与验证
KVC之-setValue:forKey:方法实现原理与验证
- (void)setValue:(id)value forKey:(NSString *)key
方法,实现原理与验证
功能:使用一个字符串标示符给一个对象的属性赋值.它支持普通对象和集合对象
这个方法的默认实现如下:
(1).首先去接收者(调用方法的那个对象)的类中查找与key相匹配的访问器方法(-set<Key>
),如果找到了一个方法,就检查它参数的类型,如果它的参数类型不是一个对象指针类型,但是只为nil
,就会执行setNilValueForKey:
方法,setNilValueForKey:
方法的默认实现,是产生一个NSInvalidArgumentException
的异常,但是你可以重写这个方法.如果方法参数的类是一个对象指针类型,就会简单的执行这个方法,传入对应的参数.如果方法的参数类型是NSNumber
或NSValue
的对应的基本类型,先把它转换为基本数据类,再执行方法,传入转换后的数据.**
(2).如果没有对应的访问器方法(setter
方法),如果接受者的类的+accessInstanceVariablesDirectly
方法返回YES
,那么就查找这个接受者的与key
相匹配的实例变量(匹配模式为_<key>
,_is<Key>
,<key>
,is<Key>
):比如:key
为age
,只要属性存在_age
,_isAge
,age
,isAge
中的其中一个就认为匹配上了,如果找到这样的一个实例变量,并且的类型是一个对象指针类型,首先released
对象上的旧值,然后把传入的新值retain
后的传入的值赋值该成员变量,如果方法的参数类型是NSNumber
或NSValue
的对应的基本类型,先把它转换为基本数据类,再执行方法,传入转换后的数据.
(3).如果访问器方法和实例变量都没有找到,执行setValue:forUndefinedKey:
方法,该方法的默认实现是产生一个 NSUndefinedKeyException
类型的异常,但是我们可以重写setValue:forUndefinedKey:方法
验证:
定义一个Person类:如下
@interface Person : NSObject
{
NSString *_name;
int _age;
NSString *_address;
}
@property (nonatomic, copy) NSString *name;
@property (nonatomic,assign) int age;
@end
@implementation Person
-(void)setName:(NSString *)name
{
NSLog(@"%s----------%@",__func__,name);
_name = name;
}
- (void) setAge:(int)age
{
_age = age;
NSLog(@"%s------%d",__func__,age);
}
- (int) age
{
NSLog(@"%s------%d",__func__,_age);
return _age;
}
- (NSString *) name
{
NSLog(@"%s----------%@",__func__,_name);
return _name;
}
@end
测试代码
1)验证: setValue:forKey:确实会调用-set<Key>方法
Person *p = [[Person alloc] init];
[p setValue:@"小明" forKey:@"name"];
输出结果
2015-08-15 20:56:56.975 company[1254:98490] -[Person setName:]----------小明
2015-08-15 20:56:56.975 company[1254:98490] -[Person setAge:]------10
2)验证:如果它的参数类型不是一个对象指针类型,但是只为nil
,就会执行setNilValueForKey:
方法,setNilValueForKey:
方法的默认实现,是产生一个NSInvalidArgumentException
的异常
测试代码[p setValue:nil forKey:@"age"];
运行结果:
2015-08-15 20:59:36.111 company[1300:100841]
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason:
3)可以重写这个方法setNilValueForKey:
在Person类的实现文件中,重写setNilValueForKey:
- (void) setNilValueForKey:(NSString *)key
{
NSLog(@"%s",__func__);
}
`
再次运行,结果:
2015-08-15 21:29:21.167 company[528:6226] -[Person setNilValueForKey:]
4)验证如果方法的参数类型是NSNumber
或NSValue
的对应的基本类型,先把它转换为基本数据类,再执行方法,传入转换后的数据,测试代码
Person.m 文件中:
- (void) setAge:(int)age
{
_age = age;
NSLog(@"%s------%d",__func__,age);
}
测试方法中
[p setValue:@(10) forKey:@"age"];
执行结果
2015-08-15 21:54:23.477 company[607:15602] -[Person setAge:]------10
5)验证如果如果没有对应的访问器方法(setter
方法),如果接受者的类的+accessInstanceVariablesDirectly
方法返回YES
,那么就查找这个接受者的与key
相匹配的实例变量(匹配模式为_<key>
,_is<Key>
,<key>
,is<Key>
):比如:key
为age
,只要属性存在_age
,_isAge
,age
,isAge
中的其中一个就认为匹配上了,如果找到这样的一个实例变量,并且的类型是一个对象指针类型,首先released
对象上的旧值,然后把传入的新值retain
后的传入的值赋值该成员变量,如果方法的参数类型是NSNumber
或NSValue
的对应的基本类型,先把它转换为基本数据类,再执行方法,传入转换后的数据.
验证:+accessInstanceVariablesDirectly默认返回YES
测试代码
NSLog(@"%d",[Person accessInstanceVariablesDirectly]);
输出结果:
2015-08-15 22:05:22.646 company[782:21098] 1
在Person
类中分别使用
@interface Person : NSObject
{
// NSString *address;
// NSString *_address;
// 注意is后面第一个字母必须大写否则会产生NSUnknownKeyException异常
// NSString *isAddress;
NSString *_isAddress;
}
测试代码
NSLog(@"%d",[Person accessInstanceVariablesDirectly]);
[p setValue:@"金燕龙大厦" forKey:@"address"];
NSString *address = [p valueForKey:@"address"];
输出结果:
2015-08-15 22:05:22.646 company[782:21098] 金燕龙大厦
6)验证:如果访问器方法和实例变量都没有找到,执行setValue:forUndefinedKey:
方法,该方法的默认实现是产生一个 NSUndefinedKeyException
类型的异常,但是我们可以重写setValue:forUndefinedKey:
方法
测试代码:
[p setValue:@"美女" forKey:@"老婆"];
结果产生一个NSUnknownKeyException:
Terminating app due to uncaught exception 'NSUnknownKeyException', reason: '[<Person 0x7fd0394a4c10> setValue:forUndefinedKey:]: this class is not key value coding-compliant for the key 老婆.'
在Person.m
文件中重写 - (void)setValue:(id)value forUndefinedKey:(NSString *)key
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
{
NSLog(@"%s",__func__);
NSLog(@"%@=%@",key,value);
}
再次运行程序输出结果:
2015-08-15 22:14:19.866 company[885:25268] -[Person setValue:forUndefinedKey:]
2015-08-15 22:14:19.866 company[885:25268] 老婆=美女
KVC之-setValue:forKey:方法实现原理与验证的更多相关文章
- setObject:forKey和setValue:forKey的区别
setObject:forKey: 是NSMutableDictionary类的方法 key参数类型可以是任意类型对象 ...
- 从[id setValue: forKey:]了解KVC
<Objective-C基础教程> P224页有详细介绍 下边是apple官网的简单介绍 和一个应用的例子. KVC就是Key-value coding,大意是允许通过一个Key来读写一个 ...
- iOS.KVC.setValue:forKey:
Foundation Framework 定义了 NSObject(NSKeyValueCoding), - (void)setValue:(id)value forKey:(NSString *)k ...
- KVC之-(id)valueForKey:(NSString *)key的实现原理与验证
KVC之-(id)valueForKey:(NSString *)key的实现原理与验证 2.-(id)valueForKey:(NSString *)key的实现原理与验证; #功能:使用一个字符串 ...
- Objective-c setObject:forKey:和setValue:forKey:的区别
setObject:forKey: 是NSMutableDictionary类的方法 key参数类型可以是任意类型对象 ...
- 字体属性设置(一):谷歌浏览器12px以下字体的显示;方法和原理
前言: chrome 谷歌浏览器默认的字体大小为16px:可以通过设置font-size来设置字体大小但是当设置到12px以下的时候字体大小不再改变:对于想设置其他大小的字体就很头疼,本文参考网上的方 ...
- Android ListView实现不同item的方法和原理分析
ListView实现不同item的方法和原理分析 一问题抛出Listview是android里面的重要组件,用来显示一个竖向列表,这个没有什么问题:但是有个时候列表里面的item不是一样的,如下图,列 ...
- JS跨域方法及原理
JS跨域分析判断 JS跨域:在不同域之间,JS进行数据传输或通信.比如ajax向不同的域请求数据.JS获取iframe中的页面中的值(iframe内外不同域) 只要协议.端口.域名有一个不同则 ...
- android中获取root权限的方法以及原理(转)
一. 概述 本文介绍了android中获取root权限的方法以及原理,让大家对android 玩家中常说的“越狱”有一个更深层次的认识. 二. Root 的介绍 1. Root 的目的 可以让我们拥有 ...
随机推荐
- Signavio
Signavio web建模器 从version 4.1开始,jBPM绑定了一个完全开源的基于web的BPM设计器工具 叫做'Signavio'.这个Signavio web建模器是和JBoss jB ...
- Android Parcel对象详解
关于Parcel的使用 在分析Parcel之前,首先按照分析流程,介绍下关于Parcel的相关常规使用. 首先是关于Parcel的获取: Parcel parcle = Parcel.Obtain() ...
- (原)未能启用约束。一行或多行中包含违反非空、唯一或外键约束的值与DATEADD
SQLServer2014,查询分析器中 这样的脚本是没有问题的:AND TPO.CREATEON <= DATEADD(DAY, 1, '2017/3/3 0:00:00') 但.NET D ...
- 如何在sublime3项目设置中设置python模块的搜索路径?ImportError: No module named *的解决办法
问题:之前使用pycharm,用的挺溜,但是电脑配置实在不争气,pycharm启动久了,耗去大量内存,导致运行起来越来越慢,于是转向使用sublime text. 把项目从pycharm切换到subl ...
- 各种语言性能(CPU密集型程序)比较
都进行Fib数列计算,计算到n=40的计算时间: 注意:开始,我以为上图中的第二列就是代表C++的性能.但是现在发现,完全不正确. 如果你使用同样的抽象和同样的逻辑去实现同样的代码,C和C++的性能几 ...
- hive的rownumber()的使用
举个简单的栗子: 找到最小日期的那一条记录 select * from ( select *,row_number() over (partition by id order by cast(date ...
- 云计算之路-试用Azure:制作虚拟机自定义镜像
虚拟机自定义镜像(Image)是一个很有用的功能,可以在一台虚拟机上配置好基本的系统环境,然后做个镜像,以后创建虚拟机直接从这个镜像创建,会省掉很多重复的配置工作. 阿里云与UCloud都有这个功能, ...
- 【转】Android之Adapter用法总结
1.概念 Adapter是连接后端数据和前端显示的适配器接口,是数据和UI(View)之间一个重要的纽带.在常见的View(ListView,GridView)等地方都需要用到Adapter.如下图直 ...
- SQL Server中按照条件随机返回数据
需求:查询对应关键字的数据,并随机返回一条. 这时,需要一个SQL的那只方法:NEWID(). 用法: [sql]SELECT TOP 1 * FROM Table WHERE TID = 1 ORD ...
- npm显示已安装的包
npm list -g --depth 0 显示安装过的npm包!