OC对象 - KVC

OC对象 - KVC

俗称"键值编码",可以通过一个key来访问某个属性

1. 常用API

    • (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
    • (void)setValue:(id)value forKey:(NSString *)key;
    • (id)valueForKeyPath:(NSString *)keyPath;
    • (id)valueForKey:(NSString *)key;

2. 原理

2.1 setValue:forKey:的原理

  • 在执行setValue:forKey:的时候
  • 按照setKey_setKey:顺序查找方法
  • 找到方法就传递参数,调用方法
  • 没找到方法,会看一下 accessInstanceVariablesDirectly方法的 return(默认return YES)
  • 如果 accessInstanceVariablesDirectly方法的 return NO,就会走setValue:forUndefinedKey:,并抛出异常NSUnknownKeyException
  • 否则的话
  • 此时会按照_key、_isKey、key、isKey的顺序查找成员变量,找到了就直接赋值

2.2 验证setValue:forKey:

2.2.1 查找方法阶段

  • 我们在ZSXPerson.m中实现下这两个方法
objectivec 复制代码
@implementation ZSXPerson

- (void)setAge:(int)age {
    NSLog(@"%s", __func__);
    _age = age;
}

- (void)_setAge:(int)age {
    NSLog(@"%s", __func__);
    _age = age;
}

@end

此时可以看到执行的是setAge:方法

  • 验证查找_setAge:方法

要验证_setAge:方法,我们不能再用@property 来声明 age 属性,因为@property 会自动帮我们生成 get、set 方法

  • ZSXPerson 我们去掉@property声明的age属性,.m文件里面实现_setAge:方法,然后同样使用setValue:forKey:来给 age 赋值
less 复制代码
@interface ZSXPerson : NSObject	

@end

@implementation ZSXPerson

//- (void)setAge:(int)age {
//    NSLog(@"%s", __func__);
//}

- (void)_setAge:(int)age {
    NSLog(@"%s", __func__);
}

@end
  • 我们发现_setAge:方法确实执行了
  • 如果我们把setAge:方法再打开,会发现setAge:是比_setAge:方法优先被查找调用的

2.2.2 查找成员变量阶段

2.2.2.1 验证成员变量按顺序查找
  • 我们假设ZSXPerson有这样几个成员变量。(setAge:(int)age_setAge:(int)age需要去掉,否则会优先找方法)
objectivec 复制代码
@interface ZSXPerson : NSObject	{
    @public
    int _age;
    int _isAge;
    int age;
    int isAge;
}

@end

@implementation ZSXPerson

//- (void)setAge:(int)age {
//    NSLog(@"%s", __func__);
//}

//- (void)_setAge:(int)age {
//    NSLog(@"%s", __func__);
//}

@end
  • _age

  • _isAge

  • age
  • isAge

验证了 成员变量按顺序查找

2.3 setValue:forKey: 触发KVO监听

有set方法

我们清楚KVO的原理,以及KVCsetValue:forKey:原理,已经能确定是会触发KVO监听的

没有set方法

我们主要验证下没有set方法的情况下,是否会触发KVO监听

  • 我们刚刚ZSXPerson的.m文件中,已经注释掉相应的set方法
  • 这时候给ZSXPerson的实例对象添加KVO监听 age 值变化

可以看到 KVO 监听也是会触发的

我们已经没有任何 set 方法了,但是 KVO 监听依然触发了。这是因为 KVC 内部本身会触发 keyPath 值变化通知

2.2.2.2 accessInstanceVariablesDirectly

我们尝试accessInstanceVariablesDirectly返回NO

此时它不会查找成员变量,因此找不到对应的 key 赋值,抛异常了

2.3 valueForKey:的原理

  • 在执行valueForKey:的时候
  • 按照getKeykeyisKey_key顺序查找方法
  • 找到方法就传递参数,调用方法
  • 没找到方法,会看一下 accessInstanceVariablesDirectly方法的 return(默认return YES)
  • 如果 accessInstanceVariablesDirectly方法的 return NO,就会走setValue:forUndefinedKey:,并抛出异常NSUnknownKeyException
  • 否则的话
  • 此时会按照_key、_isKey、key、isKey的顺序查找成员变量,找到了就直接赋值

2.4 验证valueForKey:

2.4.1 查找方法阶段

  • 分别创建getKeykeyisKey_key四个方法
  • 我们直接给实例对象person的age赋值,然后使用valueForKey:获取age

valueForKey:按顺序查找方法,最后查不到对应方法抛异常了

2.4.1 查找成员变量阶段

找不到方法,接下来按顺序查找成员变量

  • 我们创建_key、_isKey、key、isKey几个成员变量
  • 先给几个成员变量赋不同的值,然后使用valueForKey:获取age

valueForKey:按顺序查找成员变量,最后查不到对应方法抛异常了

@oubijiexi

相关推荐
A星空12336 分钟前
一、Linux嵌入式的I2C驱动开发
linux·c++·驱动开发·i2c
释怀不想释怀41 分钟前
Linux环境变量
linux·运维·服务器
zzzsde1 小时前
【Linux】进程(4):进程优先级&&调度队列
linux·运维·服务器
凡人叶枫1 小时前
C++中智能指针详解(Linux实战版)| 彻底解决内存泄漏,新手也能吃透
java·linux·c语言·开发语言·c++·嵌入式开发
yuanmenghao1 小时前
Linux 性能实战 | 第 7 篇 CPU 核心负载与调度器概念
linux·网络·性能优化·unix
qq_297574672 小时前
Linux 服务器 Java 开发环境搭建保姆级教程
java·linux·服务器
70asunflower2 小时前
Emulation,Simulation,Virtualization,Imitation 的区别?
linux·docker
神梦流3 小时前
GE 引擎的内存优化终局:静态生命周期分析指导下的内存分配与复用策略
linux·运维·服务器
凡人叶枫3 小时前
C++中输入、输出和文件操作详解(Linux实战版)| 从基础到项目落地,避坑指南
linux·服务器·c语言·开发语言·c++
wdfk_prog3 小时前
[Linux]学习笔记系列 -- [drivers][input]serio
linux·笔记·学习