面试搜集iOS底层知识

KVC的底层原理,使用和自定义KVC

2018-01-18  本文已影响33人  梁炜东

Kvc简介:

KVC(Key-value coding)键值编码,顾名思义。额,简单来说,是可以通过对象属性名称(Key)直接给属性值(value)编码(coding)“编码”可以理解为“赋值”。这样可以免去我们调用getter和setter方法,从而简化我们的代码,也可以用来修改系统控件内部属性(这个黑魔法且用且珍惜)。

举例:


image.png

如果不使用kvc我们正常使用是:


image.png

使用kvc的话是:


image.png

上面介绍了基本的kvc使用方法,下面我们来学习一下他的原理:

kvc取值的底层原理

引入:
假如我们使用kvc取值,


image.png

这时候运行的话肯定会报错,回报未定义的崩溃错误,那么我们接下来学习kvc他是怎么找的:
比如我们的


image.png
1》我们kvc在底层可不是仅仅找name这个key,他一共会找四个key,四个key的分别是:
_name _isName name isName
这四个key的优先级是从高到低依次是_name _isName name isName

如果这四个都没有那么就会遇到上面说的崩溃报错了
2》你以为到这里就结束了吗?
其实这个valueForKey:方法 不止找这几个成员变量,他还会找下面这个方法:


image.png
如果返回值是YES,他才去找那四个成员变量,如果是NO的话就不会去找那四个成员变量(即使定义的有),直接崩溃错误。
3》 你以为到这里又结束了吗?
这个valueForKey:方法其实还会找其他东西的,拿他去找什么的,他其实还会找getter方法(getter是个复数),会找那些get方法呢,如下:
image.png
这三个方法的优先级是:name,isName,getName
4》到此还没有完,他还会找两个方法,如下:
image.png
如果他找了这两个方法,会输出什么呢,会输出个数组,这个数组里面是10个字符串“haha”
5》,上面的还不完整,其实还可能会走一个方法。
即如果走到了2》,但是2》返回值为no,就自己饿会崩溃,如果我们不想崩溃的话,就重写这个方法
image.png

至此算是全部结束,你会疑问1》 2》 3》 4》这么多kvc都要去找,那么具体顺序是什么呢?下面我们来总结一下这个具体顺序

关于KVC valueForKey:key 的调用查找顺序
|-先调用“相关”方法,先后顺序是:
1》|-getter方法,getKey -> key -> isKey (注意字母大小写)
2》|- NSArray方法:countOfKey 和objectInKeyAtIndex
|-如果没有相关方法,会找2》也就是看+(BOOL)accessInstanceVariablesDirectly 返回值
1 》 |-YES 的话去找成员变量1》,顺序依次是_key -> _isKey -> key -> isKey
2 》 |-NO 的话就直接抛崩溃异常了。但是苹果还给了我们一个救火的方法,如果这个返回值是no,但是我们还不想崩溃怎么办,我们就只能重写valueForUndefinedKey,返回值为nil,这样就不会崩溃了

OK,到这里我们总结了关于kvc取值查找的过程,算了还是在用语言描述一下加深下印象吧,

文字总结:我们通过kvc的valueForKey:key方法去取值的时候,首先我们回去找getter的那三个方法,找到的话按照优先级得到值,如果三个getter方法没有话的,kvc会继续找NSArray方法,就是countOfKey 和objectInKeyAtIndex 他们两个,如果有的话就返回个数组。如果这两个NSArray方法也没有的话,我们会继续去找,继续去找+(BOOL)accessInstanceVariablesDirectly这个返回值为bool类型的方法,如果他的返回值是yes,那么我们就去找哪四个成员变量,也是按照优先级取值;如果+(BOOL)accessInstanceVariablesDirectly这个返回值为no的话,就不去找那四个成员变量了,回去找看有没有重载valueForUndefinedKey这个方法,按照要求重载了的话不会崩溃,kvc这个valueForKey:key会得到一个null值,如果没有重载这个方法,那么就会直接崩溃

ps:对于valueForKeyPath:path,这种点点点下去的路径key,和其他的原理一样


kvc设值的底层原理

[model setValue:@"kvcName" forKeyPath:@"name"];
通过我们上面kvc的取值的学习,这里kvc的设值也是需要找的,那她找的顺序是什么呢?

  1. 1.setter相关方法


    image.png

    setter的方法就这两个,他们的优先级为setName -> setIsName

  2. 找成员变量_name -> _isName -> name -> isName
  3. 和取值一样也要看+(BOOL)accessInstanceVariablesDirectly这个方法的返回值
  4. 去取值一样也有个救火的方法:


    image.png
  5. 如果我们给一个基本数据类型通过kvc设值,value给了一个nil,那么直接就会崩了,所以还有个异常处理方法来防治这个崩溃


    image.png

至此kvc的设值就结束了,我们来总结一下他的过程

  1. 寻找setter的两个方法
  2. 如果setter的两个方法都没有,看+(BOOL)accessInstanceVariablesDirectly这个方法的返回值方法的返回值

YES 去找四个成员变量
NO 去找valueForUndefinedKey,有的话不会崩溃,没有的话会崩溃

  1. 对于基本数据类型的setValue:value forKey:key,如果value为nil的话他会直接崩溃,所以kvc在基本数据类型value为nil的时候还回去找setNilValueForKey这个方法是否重载实现了,如果重载了就不会崩溃,否则会崩溃

自定义一个kvc, 不使用系统的kvc

  1. 首先我们想一下系统的kvc是不是任何类都可以使用,所以系统的kvc是不是肯定是写在NSObject的分类里面呢,所以我们自定的话就先新建一个NSObject的分类

  2. 我们自定义取值和设值的方法


    image.png
  3. 来实现这两个方法,先看实现set的方法


    image.png
    image.png
    image.png
    image.png

    写的时候可以自己通过自定的Person类和vc中测试一下
    下面是自己测试的部分代码:


    image.png
    image.png
    image.png

自己自定义的kvc设置值的方法,其实也就是按照上面学的逻辑来写的,kvc取值的就不在写了,比较简单,就不在这里重复了

上一篇下一篇

猜你喜欢

热点阅读