最近在做一个项目,项目收尾过程遇到点小问题,而这些问题中大部分原因是由于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是要展示的属性名。
结语###
以上是这次项目遇到的比较集中的一块,如果有什么更好的方法或者有什么写的不对的地方欢迎大家指正。