美文网首页
iOS KVC本质探索

iOS KVC本质探索

作者: 木子雨廷t | 来源:发表于2020-01-08 22:19 被阅读0次
    上篇文章讲完了KVO这篇来看看KVC,KVC在开发中的使用也算是挺多的,下面从几个方面来看看KVC
    • 常见的API用法
    • KVC工作中使用场景
    • 配合 KVO使用
    • KVC赋值原理
    • KVC取值原理
    • 通过KVC修改属性值,会不会触发KVO
    • 通过KVC修改成员变量值,会不会触发KVO
    • 字典setValue 和setObject区别
    KVC的全称是Key-Value Coding,俗称“键值编码”,可以通过一个key来访问某个属性
    常见的API用法

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

    新建两个类GoodsModel和GoodsDetailModel里面代码实现如下

    #import <Foundation/Foundation.h>
    #import "GoodsDetailModel.h"
    
    NS_ASSUME_NONNULL_BEGIN
    
    @interface GoodsModel : NSObject
    
    /*** 图片url ****/
    @property(nonatomic,copy) NSString *imageUrl;
    /*** 商品详情 ****/
    @property(nonatomic,strong) GoodsDetailModel *model;
    @end
    
    NS_ASSUME_NONNULL_END
    
    #import <Foundation/Foundation.h>
    
    NS_ASSUME_NONNULL_BEGIN
    
    @interface GoodsDetailModel : NSObject
    
    /*** 商品描述 ****/
    @property(nonatomic,copy) NSString *goodsDetail;
    
    @end
    
    NS_ASSUME_NONNULL_END
    

    设值GoodsModel 的imageUrl

    GoodsModel *model = [[GoodsModel alloc]init];
    [model setValue:@"https://www.jianshu.com/p/be23b5e46e07" forKey:@"imageUrl"];
    [model setValue:@"https://www.jianshu.com/p/be23b5e46e07" forKeyPath:@"imageUrl"];
    

    上述两行代码都能设值成功

    设置GoodsDetailModel 的goodsDetail属性

    model.model = [[GoodsDetailModel alloc]init];
    [model.model setValue:@"商品描述" forKey:@"goodsDetail"];
    [model.model setValue:@"商品描述" forKeyPath:@"model.goodsDetail"];
    

    只有第二行可以设值成功, setValue@"" forKey@""只能设值第一层的属性,而setValue@"" forKeyPath@"" Path就是路径可以一级一级的寻找就行赋值或者取值操作

    -(id)valueForKeyPath:(NSString *)keyPath;
    -(id)valueForKey:(NSString *)key; 
    

    取值方法和赋值方法类似

    KVC工作中使用场景

    1. UI操作

    [textField setValue:[UIColor grayColor] forKeyPath:@"_placeholderLabel.textColor"];
    

    自定义tabbar: 可以自定义一个UITabbar对象,然后在内部创建自己想要的视图,并通过layoutSubviews方法在内部进行重新布局。然后通过KVC的方式,将UITabbarController的tabbar属性替换为自定义的类即可。

    自定义tabbar.png

    2. 字典转model在项目中经常会遇到字典转模型的情况,如果在自定义的init方法里逐个赋值,这样每次数据发生改变还需要改赋值语句。然而通过KVC为我们提供的赋值API,可以对数据进行批量赋值。假设有以下JSON数据并定义Status类,在外界通过setValuesForKeysWithDictionary:方法对Status进行赋值。

    Status *status = [[self alloc] init];
    //利用KVC字典转模型
    [status setValuesForKeysWithDictionary:dict];    
    return status;
    

    3. 配合 KVO使用:根据KVO的实现原理是在运行时生成新的子类并重写其setter方法,在其内容发生改变时发送消息。但这只是对属性直接进行赋值会触发,如果属性是容器对象,对容器对象进行add或remove操作,则不会调用KVO的方法。可以通过KVC对应的API来配合使用,使容器对象内部发生改变时也能触发KVO。 在进行容器对象操作时,先调用下面方法通过key或者keyPath获取集合对象,然后再对容器对象进行add或remove等操作时,就会触发KVO的消息通知了。
    Key方法

    - (NSMutableArray *)mutableArrayValueForKey:(NSString *)key;
    - (NSMutableOrderedSet *)mutableOrderedSetValueForKey:(NSString *)key API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
    - (NSMutableSet *)mutableSetValueForKey:(NSString *)key;
    

    KeyPath方法

    - (NSMutableArray *)mutableArrayValueForKeyPath:(NSString *)keyPath;
    - (NSMutableOrderedSet *)mutableOrderedSetValueForKeyPath:(NSString *)keyPath API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
    - (NSMutableSet *)mutableSetValueForKeyPath:(NSString *)keyPath;
    
    KVC赋值原理

    1.按照setKey:,_setKey:顺序查找方法,找到了就调用方法传递参数。
    2.第一步没找到就会调用accessInstanceVariablesDirectly方法,该方法返回值为NO时直接调用setValue:forUndefinedKey:并抛出异常NSUnknownKeyException,方法返回值是YES的时候进入第三步。该方法默认值是返回YES。
    3.按照_key、_isKey、key、isKey顺序查找成员变量,找到了就直接赋值,没找到依然是调用setValue:forUndefinedKey:并抛出异常NSUnknownKeyException。

    图示:


    KVC赋值原理.png
    KVC取值原理

    1.kvc取值按照 getKey、key、iskey、_key 顺序查找方法
    存在直接调用
    2.没找到同样,先查看accessInstanceVariablesDirectly方法
    如果return YES; > 可以直接访问成员变量
    如果return NO; > 不可以直接访问成员变量,
    3.如果可以访问会按照 _key、_isKey、key、iskey的顺序查找成员变量
    找到直接复制
    未找到报错NSUnkonwKeyException错误

    KVC取值图示


    KVC取值原理.png
    通过KVC修改属性值,会不会触发KVO

    会触发 因为KVC也是走了setKey方法 。

    通过KVC修改成员变量值,会不会触发KVO

    会触发 KVC内部会调用willChangeValueForKey和didChangeValueForKey方法

    字典setValue 和setObject区别

    setValue forKey :
    1、方法是KVC(键-值编码),方法是在NSObject对象中创建的,也就是说所有的oc对象都有这个方法,所以可以 用于任何类

    例子:
    People *p1 = [[People alloc] init];
    [p1 setValue:@"mkj" forKey:@"name"];
    当对象有name属性的时候就是通过KVC来赋值
    2、value可以是nil,如果为nil,则自动调用removeObject forkey方法
    3、key必须为string类型的
    4、取值的时候valueforkey的key如果包含@符号,则取值时会自动把@去掉,程序crash

    setObject forKey:
    1、方法是NSMutabledictionary特有的;
    2、value不可以为nil,如果为nil,则程序崩溃在此;但value可以为[NSNull null],[NSNull null]为一个空对象,并不是nil;
    3、Key的对象是一个id类型,并不是NSString,只不过我们经常使用NSString而已。
    4、取值的时候objectforkey的key如果包含@符号,无影响,可以正常取出该值

    特别推荐:
    iOS OC对象的本质窥探(一)
    iOS OC对象的本质窥探(对象分类)(二)
    iOSKVO本质探索
    iOS获取手机唯一标示
    iOS 高德地图实现大头针展示,分级大头针,自定制大头针,在地图上画线,线和点共存,路线规划(驾车路线规划),路线导航,等一些常见的使用场景
    堆和栈详解---转载
    我的简书主页
    我的博客主页
    底层原理篇

    相关文章

      网友评论

          本文标题:iOS KVC本质探索

          本文链接:https://www.haomeiwen.com/subject/hwqsactx.html