Objective-C - KVC 是如何工作的?

Objective-C - How does KVC work under the hood?

我正在使用 KVC,我想知道它是如何工作的。

由于它使用键和值,我可以假设每个对象的 self 都包含一个包含 keys = properties namevalues = setters 的字典。因此,当我调用 [self setValue:aValue forKey:@"aProperty"] 时,对象从指向 setter 方法的键中获取哈希值。

当然,所有这些都是猜测,以及我将如何利用我的知识来实现​​它。

这就是它的工作原理吗?

从概念上讲,您的方向是正确的,尽管有点复杂。

首先,请记住 Objective-C 是一种动态语言。本质上,每个 class 维护方法名称到它们的实际实现(只是 C 函数)的映射。这类似于您所说的字典,尽管它由 Objective-C 运行时本身管理。

每次调用方法时都会使用此映射。例如,假设您有这段代码:

[obj doSomething];

真正发生的是,在运行时,Objective-C 运行时搜索 obj 的方法映射以查找名为 "doSomething" 的条目。这个 returns 一个 函数 运行时调用,将 obj 作为第一个参数传递给该函数。

因为方法是在运行时分派的,Objective-C 提供了多种使用字符串调用函数的方法。 (这类似于在 Python 中执行类似 getattr(obj, "doSomething")() 的操作,如果您熟悉 Python。)

Objective-C 运行时本身也会跟踪实例变量的名称以及它们在内存中相对于对象的实际位置。

这就是 KVC 能够完成任务的方式。当你打电话时:

[obj setValue:@"value" forKey:@"property"];

KVC 运行时使用 Objective-C 运行时首先查找名为 setProperty 的方法。运行时获取与该方法对应的函数,然后 KVC 机制可以调用该方法,将 obj@"value" 作为参数传递给该函数。

找不到方法怎么办?好吧,然后 KVC 机器寻找具有相同名称的 实例变量 ,使用来自 Objective-C 运行时的函数,如 ivar_getOffset or the like. Probably at some point it uses a function like object_setIvar 来设置实例变量。 (这是推测,但我认为这是关于 KVC 如何处理实例变量的一个很好的猜测。)

如果 KVC 找不到方法和实例变量,它会调用 setValue:forUndefinedKey: or valueForUndefinedKey:,可以选择在 class 上定义它以动态处理属性。


简而言之,您的想法是正确的,但是 属性 名称到方法(或 ivars)的映射是由 Objective-C 运行时完成的,这可能要归功于Objective-C.


另请注意,KVC 的工作方式相当简单。 KVO 有点复杂。