Runtime应用之KVC

runtime可以以底层的角度来对一些实现方式进行更改,比如说KVC

首先,先来了解下KVC的底层原理:

key : value

  • 1.去模型中查找有没有setValue:,直接调用这个对象setValue:赋值
  • 2.如果没有setValue:,就在模型中查找_value属性
  • 3.如果没有_value属性,就查找value属性
  • 4.如果还没有就报错

在和后台通信的JSON数据中,可能会看到这种JSON数据:

{
    "id" : "tripleCC",
    "age" : "30",
    "address" : "杭州",
    "schooll" : "HDU"
    ...
}

其中的id是什么?是Objective-C关键字,也就是说我定义以下属性会出现警告:

@property (nonatomic, strong) NSString *id;

虽然可以使用以下方法,对模型中的成员变量进行统一设置,但是出现警告总归是不好的:

- (void)setValuesForKeysWithDictionary:(NSDictionary *)keyedValues;

既然这样,可以选择手动一个个去实现。但是这样在数据少的时候可以试试,在数据比较多时就不太现实了,程序的可扩展性也不好。

现在来了解下相对比较简单的两种解决方法:

方式1.重写setValue:forKey:

setValuesForKeysWithDictionary:的底层是调用setValue:forKey:的,所以可以考虑重写这个方法,并且判断其key是id时,手动转换成模型的成员变量名,这里假设把id对应成以下属性:

@property (nonatomic, strong) NSString *ID;

有了对应的属性名后,就可以重写底层方法了:

  • 如下所示,当判断到key的值为id时,我手动将key转换成了模型属性名,即ID
- (void)setValue:(id)value forKey:(NSString *)key
{
    if ([key isEqualToString:@"id"]) {
        [self setValue:value forKeyPath:@"ID"];
    }else{
        [super setValue:value forKey:key];

    }
}

这样,当使用setValuesForKeysWithDictionary:就不会出现模型中找不到对应的成员变量的错误了。

方式2.使用runtime

考虑到runtime和KVC的实现原理,可以使用另一种实现思路,就是先在模型中找到对应的成员变量,然后从JSON字典中找到对应的数据进行赋值

这里先要了解runtime的两个实例变量操作方法:

// 获取成员变量列表
Ivar * class_copyIvarList ( Class cls, unsigned int *outCount );
// 获取成员变量名
const char * ivar_getName ( Ivar v );

详细实现步骤:

  • 1.获取模型中的所有实例变量

    unsigned int outCount = 0;
    Ivar *ivars = class_copyIvarList(self, &outCount);
    
  • 2.将获取出来以''开头的实例变量名去处''符号

    NSString *ivarName = @(ivar_getName(ivar));
    ivarName = [ivarName substringFromIndex:1];
    
  • 3.获取JOSN字典中对应的value,如果没有,手动设置我们传入的字典映射,以指定对应的模型变量名,最后调用setValue:forKeyPath:设置模型实例变量值

    id value = dict[ivarName];
    // 由外界通知内部,模型中成员变量名对应字典里面的哪个key
    // 这里是ID -> id
    if (value == nil) {
        // 这里的mapDict就是外界传入的映射字典
        if (mapDict) {
            NSString *keyName = mapDict[ivarName];
    
            value = dict[keyName];
        }
    }
    [objc setValue:value forKeyPath:ivarName];
    

由于需要针对所有模型使用,可以将其设置为NSObject分类。以上步骤的完整代码为:

// dict  -> 资源文件提供的字典
// mapDict  -> 提供的key映射(实际变量名:资源文件key)
+ (instancetype)objcWithDict:(NSDictionary *)dict mapDict:(NSDictionary *)mapDict
{
    id objc = [[self alloc] init];


    // 遍历模型中成员变量
    unsigned int outCount = 0;
    Ivar *ivars = class_copyIvarList(self, &outCount);

    for (int i = 0 ; i < count; i++) {
        Ivar ivar = ivars[i];

        // 成员变量名称
        NSString *ivarName = @(ivar_getName(ivar));

        // 获取出来的是`_`开头的成员变量名,需要截取`_`之后的字符串
        ivarName = [ivarName substringFromIndex:1];

        id value = dict[ivarName];
        // 由外界通知内部,模型中成员变量名对应字典里面的哪个key
        // ID -> id
        if (value == nil) {
            if (mapDict) {
                NSString *keyName = mapDict[ivarName];

                value = dict[keyName];
            }
        }
        [objc setValue:value forKeyPath:ivarName];
    }
    return objc;
}

使用方法:


+ (instancetype)itemWithDict:(NSDictionary *)dict
{
    // 传入key和实例变量名的映射字典@{@"ID":@"id"}
    TPCItem *item = [TPCItem objcWithDict:dict mapDict:@{@"ID":@"id"}];

    return item;
}

你可能感兴趣的:(Runtime应用之KVC)