欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

NSDictionary与模型相爱相杀

程序员文章站 2022-07-14 16:44:22
...

最近在做一个项目,项目收尾过程遇到点小问题,而这些问题中大部分原因是由于NSDictionary和模型方面的问题。所以今天我针对它做个小总结。

给模型赋值###

问题的大部分根源都是在数据解析成模型数据那一块,这里不得不说映射,将Json里对应字段的内容赋值给模型对应属性

JFCustomerModel *customerModel = [[JFCustomerModel alloc] init];
[customerModel setValuesForKeysWithDictionary:dic];
customerModel.bankAccount = dic[@"bankAccount"];
customerModel.bankName = dic[@"bankName"];
customerModel.clientAddress = dic[@"clientAddress"];
customerModel.clientMobile = dic[@"clientMobile"];
customerModel.clientName = dic[@"clientName"];
customerModel.contractNo = dic[@"contractNo"];
customerModel.customerType = dic[@"customerType"];

你也许会这么做,但是我可以告诉你不用这么做,可以优化

- (NSArray *)allPropertyNames
{
    unsigned count;
    objc_property_t *properties = class_copyPropertyList([self class], &count);

    NSMutableArray *rv = [NSMutableArray array];

    unsigned i;
    for (i = 0; i < count; i++)
    {
        objc_property_t property = properties[i];
        NSString *name = [NSString stringWithUTF8String:property_getName(property)];
        [rv addObject:name];
    }

    free(properties);

    return rv;
}

- (void)setModelValueWithDictionary:(NSDictionary *)dictionary
{
    
    
    NSArray *properties = [self allPropertyNames];

    for (NSString *name in properties) {
        NSLog(@"%@",name);
        NSString * val;
        
        id obj=dictionary[name];
        if((obj == nil || [obj isKindOfClass:[NSNull class]])){
            val= @"";
        }else if([obj isKindOfClass:[NSString class]]){
            val = obj;
        }else if([obj isKindOfClass:[NSNumber class]]){
            val = [obj stringValue];
        }
        
        //dictionary[name];
        [self setValue:val forKey:name];
    }
}

这是动态获取模型的所有属性,运用了runtime,但我想说OC已经给我们封装了一个方法,而且特别简单

[customerModel setValuesForKeysWithDictionary:dic];

但是这个方法有几点需要注意的:

  • 字典的key名需要与模型的属性名一致
  • 模型里可以存在字典里不存在的属性名,但是如果字典有的,但模型里没有,就会引起程序崩溃。
    第一点的补救方式就只有一个一个对应赋值。
    关于这第二点还是有补救的,在模型里声明并覆盖这个方法,这个方法里不需要写任何东西。
-(void)setValue:(id)value forUndefinedKey:(NSString *)key;

NSDictionary解析值为NSNULL###

这种情况会导致程序崩溃,特别让人头疼,如果是个nil类型都好,可是这个NSNULL类型在未做判断的情况下使用它就会导致程序崩溃。之前还想着在显示的时候做个判断,但是发现这样修改的地方就特别多,所以想想还是在源头就给数据处理一下,可还是不想加判断,最后就直接给NSDictionary改装一下,添加一个方法,如果拿到的值时是NSNULL类型就返回@“”,写个NSDictionary的分类

@implementation NSDictionary (JFEXtension)

- (NSDictionary *)dictionaryByReplacingNullsWithBlanks {
    const NSMutableDictionary *replaced = [self mutableCopy];
    const id nul = [NSNull null];
    const NSString *blank = @"";
    
    for (NSString *key in self) {
        id object = [self objectForKey:key];
        if (object == nul) [replaced setObject:blank forKey:key];
        else if ([object isKindOfClass:[NSDictionary class]]) [replaced setObject:[object dictionaryByReplacingNullsWithBlanks] forKey:key];
    }
    return [NSDictionary dictionaryWithDictionary:[replaced copy]];
}

@end

这样解析数据前先掉一下这个方法,这里说明一下这个方法只能适用只有一层数据的,对于那些字典里套字典那种数据不适用,不过可以多次调用这个方法,关键还是要领会

模型数据换值###

接口解析出来的数据可能不是你想要的数据,还需要处理一下,比如那些1代表身份证、2代表学号······,这种情况下有两种情况

传过来的值和所要展示的值类型相同,如NSString#####

这种情况你可以重写属性的set方法

- (void)setClientName:(NSString *)clientName
{
    if ([clientName isEqualToString:@"张三"]) {
        _clientName = @"李四";
    }else if([clientName isEqualToString:@"李四"]){
        _clientName = @"王五";
    }else{
        _clientName = clientName;
    }
}
传过来的值和所要展示的值类型不相同,如所传是NSNumber类型,而展示类型却是NSString类型#####

这种情况需要重新添加一个属性,重写它的get方法

- (NSString *)tradeTypeStr
{
    if ([_tradeType isEqual:@1]) {
        _tradeTypeStr = @"认购";
    }else if([_tradeType isEqual:@2]){
        _tradeTypeStr = @"申购";
    }else if([_tradeType isEqual:@3]){
        _tradeTypeStr = @"赎回";
    }
    return _tradeTypeStr;
}

这里tradeType是解析给模型的属性名,而tradeTypeStr是要展示的属性名。

结语###

以上是这次项目遇到的比较集中的一块,如果有什么更好的方法或者有什么写的不对的地方欢迎大家指正。