iOS开发中,网络请求得到json转化为字典,然后字典转化为模型,这是很普遍要做的事。成型的第三方框架也有很多,前段时间比较火的YYKit中的YYMoel对各大这方面的框架包括JsonModel,MjExtension等效率都有所对比。但是授之以鱼不如授之以渔,有时候我们仅仅想要的就是字典转模型而已,简单,可控,可自定义。今天,小编我提供了一下自己的解决方案。
先来一波思路分析。
后台返回的数据json(xml很少人在用了吧)的类型的数据格式有对象和数组
, 字符串
,数字
,布尔
,null
,使用系统自带的NSJSONSerialization
得到字典,会将对象转化为NSDictionary对象,数组转化为NSArray对象,字符串转化为NSString对象,数字和布尔类型转化为NSValue对象或者子类NSNumber对象,null转化为NSNull对象。json中的null这个就需要小心了。java的后台程序可能是直接将model转化为json,当对象没初始化为null时,json就会出现null,而不是应该有的{}。null会转化为NSNull,但是我们认为他是对象类型,转化为了NSDictionary对象,然后调用了objectForkey,就会报unRecognized selector exception
使程序崩溃。关于这点,我曾经和做后台的同事争吵过,说:你既然定义json中某key的值是对象类型,为空你也要传“{}”啊({}会转化为空字典类型),为什么传null。他们争论到:从数据库中查不到,就没必要初始化model对象,转化为json也就会为null。我直接无语了。
直接上代码吧。注释还是蛮清晰的,记得不要忘记把那两个“安全设置”加上,不然,碰到手误,忘记定义的属性,程序又该崩溃了。setNilValueForKey:
这个不常用,是定义assign类型的属性给它set nil才会触发。加上也不多。
- (instancetype)initWithDic:(NSDictionary*)dic
{
if (!dic || ![dic isKindOfClass:[NSDictionary class]]) {
return nil;
}
if (self = [super init]) {
for (NSString *key in [dic allKeys]) {
id value = dic[key];
//1.处理对象类型和数组类型
if ([value isKindOfClass:[NSArray class]] || [value isKindOfClass:[NSDictionary class]]) {
[self setValue:value forKeyPath:key];
}
//2.处理空类型:防止出现unRecognized selector exception
else if ([value isKindOfClass:[NSNull class]]) {
// [self setValue:nil forKey:key];
}
//3.处理其他类型:包括数字,字符串,布尔,全部使用NSString来处理
else{
[self setValue:[NSString stringWithFormat:@"%@",value] forKeyPath:key];
}
}
}
return self;
}
#pragma mark KVC 安全设置
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
{
NSLog(@"%s",__func__);
}
- (void)setNilValueForKey:(NSString *)key
{
NSLog(@"%s",__func__);
}
JSONModel有一个好处,就是我们在po或者log打印model对象的时候回直接展示他的属性值。其实就是重写description
方法而已。我们也来一波自定义:
#pragma mark po或者打印时打出内部信息
-(NSString *)description
{
NSMutableString* text = [NSMutableString stringWithFormat:@"<%@> \n", [self class]];
NSArray* properties = [self filterPropertys];
[properties enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
NSString* key = (NSString*)obj;
id value = [self valueForKey:key];
NSString* valueDescription = (value)?[value description]:@"(null)";
if ( ![value respondsToSelector:@selector(count)] && [valueDescription length]>60 ) {
valueDescription = [NSString stringWithFormat:@"%@...", [valueDescription substringToIndex:59]];
}
valueDescription = [valueDescription stringByReplacingOccurrencesOfString:@"\n" withString:@"\n "];
[text appendFormat:@" [%@]: %@\n", key, valueDescription];
}];
[text appendFormat:@"</%@>", [self class]];;
return text;
}
方法调用了[self filterPropertys]
获本类的所有属性。这个用到了所谓的“高大上”的objc的runtime中方法了。先来一波包含头文件#import <objc/runtime.h>
。然后在上代码。
#pragma mark 获取一个类的属性列表
- (NSArray *)filterPropertys
{
NSMutableArray* props = [NSMutableArray array];
unsigned int count;
objc_property_t *properties = class_copyPropertyList([self class], &count);
for(int i = 0; i < count; i++){
objc_property_t property = properties[i];
const char* char_f =property_getName(property);
NSString *propertyName = [NSString stringWithUTF8String:char_f];
[props addObject:propertyName];
// NSLog(@"name:%s",property_getName(property));
// NSLog(@"attributes:%s",property_getAttributes(property));
}
free(properties);
return props;
}
最后。字典转模型,你是转了,那模型转字典呢?作为一个实用主义的程序员,如果不是有需求用到了,我才不去想这个的,多烧脑子啊😄。开发时,有时候需要提交数据给后台,由于网络请求的封装,只需要传一个字典对象过去就行。如果需要把一个model对象所有属性都作为参数提交,那么就需要吧model转化为字典类型。方法如下:
#pragma mark 模型中的字符串类型的属性转化为字典
-(NSDictionary*)modelStringPropertiesToDictionary
{
NSArray* properties = [self filterPropertys];
NSMutableDictionary* dic = [NSMutableDictionary dictionary];
[properties enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
NSString* key = (NSString*)obj;
id value = [self valueForKey:key];
if ([value isKindOfClass:[NSString class]]) {
NSString* va = (NSString*)value;
if (va.length > 0) {
[dic setObject:value forKey:key];
}
}
}];
return dic;
}
将以上代码封装一个BaseModel类,所有model类继承它。
以上只是根据自己的所学加以运用而已,一千个读者就有一千个哈姆雷特。好多东西不是没法解决,只是暂时不知道解决的办法而已。这篇文章是自己的所学的一个总结,希望对读者有所帮助。