iOS底层原理:KVC分析(一)

简介: KVC全称Key-Value Coding,俗称键值编码。它是一种通过字符串描述符而不是通过调用访问方法或者直接使用实例变量的非直接的访问对象属性的机制。在iOS中,NSObject、NSArray、NSDictionary等类使用这种机制并采用分类的形式为自身拓展了KVC的能力。

KVC全称Key-Value Coding,俗称键值编码。它是一种通过字符串描述符而不是通过调用访问方法或者直接使用实例变量的非直接的访问对象属性的机制。在iOS中,NSObject、NSArray、NSDictionary等类使用这种机制并采用分类的形式为自身拓展了KVC的能力。常用的Api如下:


- (void)setValue:(nullable id)value forKey:(NSString *)key;
- (nullable id)valueForKey:(NSString *)key;


点方法进去:


image.png

image.png

image.png


我们可以看到,它是在 Foundation 框架下,关于NSObject、NSArray、NSDictionary等的一个NSKeyValueCoding的分类。其实这些内容在苹果的官方文档中解释的很清楚,小伙伴们可以自行查阅。Key-Value Coding Programming Guide


KVC设值和取值


KVC的使用我们都不陌生,那么KVC在内部操作时是以怎样的顺序来寻找key的呢?接下来我们就来探索一下。


取值 valueForKey:

先看下苹果官方文档的Setter步骤:


image.png

image.png


国际惯例,翻译一下:


  1. 在实例中搜索找到的第一个名称为get、、is、或的访问器方法_,按该顺序。如果找到,则调用它并使用结果继续执行步骤 5。否则继续下一步。


  1. 如果没有找到简单的访问器方法,则在实例中搜索名称与模式countOfobjectInAtIndex:(对应于NSArray类定义的原始方法)和AtIndexes:(对应于NSArray方法objectsAtIndexes:)的方法。
    如果找到这些中的第一个和其他两个中的至少一个,则创建一个响应所有NSArray方法的集合代理对象并返回该对象。否则,继续执行步骤 3。
    代理对象随后将任何NSArray接收到的一些组合的消息countOfobjectInAtIndex:AtIndexes:消息给键-值编码创建它兼容的对象。如果原始对象还实现了一个可选的方法,其名称类似于get:range:,则代理对象也会在适当的时候使用它。实际上,与键值编码兼容的对象一起工作的代理对象允许底层属性表现得好像它是NSArray,即使它不是。


  1. 如果没有找到简单的访问方法或阵列访问方法组,寻找一个三重的方法命名countOfenumeratorOf和memberOf:(对应于由所定义的原始的方法NSSet类)。
    如果找到所有三个方法,则创建一个响应所有NSSet方法的集合代理对象并返回该对象。否则,继续执行步骤
    此代理对象随后将任何NSSet接收到的一些组合信息countOfenumeratorOfmemberOf:消息以创建它的对象。实际上,与键值编码兼容的对象一起工作的代理对象允许底层属性表现得好像它是NSSet,即使它不是。


  1. 如果发现收集的访问方法没有简单的存取方法或者组,如果接收器的类方法accessInstanceVariablesDirectly返回YES,搜索名为实例变量__is,,或者is,按照这个顺序。如果找到,直接获取实例变量的值并进行步骤5,否则进行步骤6。


  1. 如果检索到的属性值是一个对象指针,只需返回结果即可。
    如果该值是 支持的标量类型NSNumber,则将其存储在一个NSNumber实例中并返回该实例。如果结果是NSNumber不支持的标量类型,则转换为NSValue对象并返回。


  1. 如果所有其他方法都失败,请调用valueForUndefinedKey:,默认情况下,这会引发异常,但 的子类NSObject可能会提供特定于键的行为。

这里推荐一个翻译工具DeepL,个人觉得还是挺准挺好用的。好了,废话不多说,我们顺着这个步骤来验证一下。

首先按照文档中的顺序get ->  -> is -> _ 进行调试验证即可,如图


image.png

如果以上方法都没有,且accessInstanceVariablesDirectly为YES时,则按照_ -> _is ->  -> is取值,如图:


image.png


image.png

以上取值顺序依次注释代码进行验证即可,如果都没有的话,就会调用valueForUndefinedKey:这个方法,最终抛出异常。

设值 setValue:forKey:

先看下苹果官方文档的Setter步骤:

image.png

翻译如下:


  1. 按顺序寻找名为set:_set的第一个访问器。如果找到了,就用输入值(或根据需要解开的值)来调用它,然后完成。


  1. 如果没有找到简单的访问器,并且如果类方法accessInstanceVariablesDirectly返回YES,那么寻找一个名称为__is、或is的实例变量,依次进行。如果找到了,直接用输入值(或解开的值)设置变量,然后完成。


  1. 一旦发现没有访问器或实例变量,就调用setValue:forUndefinedKey:。这默认会引发一个异常,但NSObject的子类可以提供特定于键的行为。

首先来验证:set:  ->_set:


image.png

image.png

从上面getter取值顺序我们可以看到有get ->  -> is -> _四个方法,而此处的setter只有set:_set:两个方法。显然,对setter的第一步骤有所怀疑,那就探索一下,实现setIs_setIs,如图:


image.png


经过代码验证,这里确实会走setIs:,而_setIs:没有调用。所以苹果文档这里的描述并不全面。

set: -> _set: ->setIs都没有实现的时候,

并且当accessInstanceVariablesDirectly设置为YES时 获取实例变量的顺序为顺序查找名称为_ -> _is ->  -> is,如下图:


image.png

image.png


根据上面的流程,梳理一张setValue:forKey:的流程图:


image.png


相关文章
|
12天前
|
开发工具 Android开发 Swift
安卓与iOS开发环境对比分析
在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统无疑是主角。它们各自拥有独特的特点和优势,为开发者提供了不同的开发环境和工具。本文将深入浅出地探讨安卓和iOS开发环境的主要差异,包括开发工具、编程语言、用户界面设计、性能优化以及市场覆盖等方面,旨在帮助初学者更好地理解两大平台的开发特点,并为他们选择合适的开发路径提供参考。通过比较分析,我们将揭示不同环境下的开发实践,以及如何根据项目需求和目标受众来选择最合适的开发平台。
27 2
|
2月前
|
IDE 开发工具 Android开发
安卓与iOS开发环境对比分析
在移动应用开发的广阔舞台上,安卓和iOS无疑是两大主角。它们各自拥有独特的开发环境、工具集和生态系统。本文深入探讨了这两个平台的开发环境,从编程语言、IDE到模拟器的使用,再到市场覆盖度和用户群体差异。通过比较,旨在为开发者提供选择合适平台的参考依据,并引发对如何优化跨平台开发策略的进一步思考。
36 0
|
17天前
|
IDE 开发工具 Android开发
安卓与iOS开发环境对比分析
本文将探讨安卓和iOS这两大移动操作系统在开发环境上的差异,从工具、语言、框架到生态系统等多个角度进行比较。我们将深入了解各自的优势和劣势,并尝试为开发者提供一些实用的建议,以帮助他们根据自己的需求选择最适合的开发平台。
23 1
|
1月前
|
Java 开发工具 Android开发
安卓与iOS开发环境对比分析
【8月更文挑战第20天】在移动应用开发的广阔天地中,Android和iOS两大平台各自占据着重要的位置。本文将深入探讨这两种操作系统的开发环境,从编程语言到开发工具,从用户界面设计到性能优化,以及市场趋势对开发者选择的影响。我们旨在为读者提供一个全面的比较视角,帮助理解不同平台的优势与挑战,并为那些站在选择十字路口的开发者提供有价值的参考信息。
|
1月前
|
开发框架 Android开发 Swift
安卓与iOS应用开发对比分析
【8月更文挑战第20天】在移动应用开发的广阔天地中,安卓和iOS两大平台各占半壁江山。本文将深入探讨这两大操作系统在开发环境、编程语言、用户界面设计、性能优化及市场分布等方面的差异和特点。通过比较分析,旨在为开发者提供一个宏观的视角,帮助他们根据项目需求和目标受众选择最合适的开发平台。同时,文章还将讨论跨平台开发框架的利与弊,以及它们如何影响着移动应用的开发趋势。
|
1月前
|
安全 搜索推荐 Android开发
安卓与iOS应用开发的对比分析
【8月更文挑战第20天】在移动应用开发领域,安卓和iOS两大平台各领风骚。本文通过深入探讨两者的开发环境、编程语言、用户界面设计、应用市场及分发机制等方面的差异,揭示了各自的优势和挑战。旨在为开发者提供决策支持,同时帮助理解为何某些应用可能优先选择在一个平台上发布。
27 2
|
1月前
|
开发工具 Android开发 Swift
安卓与iOS开发环境对比分析
在移动应用开发的大舞台上,安卓与iOS两大操作系统各占半壁江山。本文将深入浅出地比较两者的开发环境,从开发工具、编程语言到用户界面设计等多个维度进行剖析,旨在为初入行的开发者们提供一盏明灯,帮助他们选择适合自己的开发路径。通过实例分析和数据支持,我们将揭示这两个平台的独特优势和潜在挑战,以及它们如何影响应用的性能和用户体验。
56 1
|
1月前
|
开发框架 开发工具 Android开发
安卓与iOS开发环境对比分析
【8月更文挑战第7天】在移动应用开发的广阔天地中,安卓和iOS两大平台各据一方。本文将深入探讨这两个平台的异同,从开发环境、工具支持到市场定位等方面进行比较。我们将不涉及具体的代码示例,而是聚焦于开发生态的宏观视角,为即将踏入这一领域的开发者提供一个全面的视角。
76 6
|
1月前
|
Android开发 Swift iOS开发
安卓与iOS开发环境的差异性分析
【8月更文挑战第6天】在移动应用开发的广阔天地中,安卓和iOS两大平台各自占据半壁江山。本文将深入探讨这两个操作系统在开发环境方面的主要差异,从编程语言、工具框架到开发成本等多个维度进行比较。通过分析,旨在为开发者提供一个清晰的平台选择指导,帮助他们根据项目需求和个人偏好做出更明智的决策。
80 5
|
1月前
|
开发工具 Android开发 Swift
安卓与iOS开发环境对比分析
在移动应用开发的广阔天地中,安卓与iOS两大平台各占半壁江山。本文通过浅显的语言和直观的比喻,探讨了这两大操作系统在开发环境上的差异与特点,旨在为初入行的开发者们提供一个清晰的指南。我们将从开发工具、编程语言、用户界面设计以及生态系统四个方面进行比较,帮助读者理解每个平台的优势与局限。