首先說下運用KVC將字典轉模型的方式:
@implementation Status
+ (instancetype)statusWithDict:(NSDictionary *)dict
{
Status *status = [[self alloc] init];
[status setValuesForKeysWithDictionary:dict];
return status;
}
@end
KVC字典轉模型弊端:必須保證,模型中的屬性和字典中的key一一對應。
- 如果不一致,就會調用
[setValue:forUndefinedKey:]
報key
找不到的錯。 - 分析:模型中的屬性和字典的key不一一對應,系統就會調用
setValue:forUndefinedKey:
報錯。 - 解決:重寫對象的
setValue:forUndefinedKey:
,把系統的方法覆蓋, 就能繼續使用KVC,字典轉模型了。
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
{
//重寫這個方法解決報錯
}
利用Runtime將字典轉模型:
demo鏈接:https://github.com/VitoJIanxue/RuntimeToModel
- 思路:利用運行時,遍歷模型中所有屬性,根據模型的屬性名,去字典中查找key,取出對應的值,給模型的屬性賦值。
- 步驟:提供一個NSObject分類,專門字典轉模型,以後所有模型都可以通過這個分類轉。
@implementation NSObject (Model)
+ (instancetype)modelWithDict:(NSDictionary *)dict
{
// 思路:遍歷模型中所有屬性-》使用運行時
// 0.創建對應的對象
id objc = [[self alloc] init];
// 1.利用runtime給對象中的成員屬性賦值
// class_copyIvarList:獲取類中的所有成員屬性
// Ivar:成員屬性的意思
// 第一個參數:表示獲取哪個類中的成員屬性
// 第二個參數:表示這個類有多少成員屬性,傳入一個Int變量地址,會自動給這個變量賦值
// 返回值Ivar *:指的是一個ivar數組,會把所有成員屬性放在一個數組中,通過返回的數組就能全部獲取到。
/* 類似下面這種寫法
Ivar ivar;
Ivar ivar1;
Ivar ivar2;
// 定義一個ivar的數組a
Ivar a[] = {ivar,ivar1,ivar2};
// 用一個Ivar *指針指向數組第一個元素
Ivar *ivarList = a;
// 根據指針訪問數組第一個元素
ivarList[0];
*/
unsigned int count;
// 獲取類中的所有成員屬性
Ivar *ivarList = class_copyIvarList(self, &count);
for (int i = 0; i < count; i++) {
// 根據角標,從數組取出對應的成員屬性
Ivar ivar = ivarList[i];
// 獲取成員屬性名
NSString *name = [NSString stringWithUTF8String:ivar_getName(ivar)];
// 處理成員屬性名->字典中的key
// 從第一個角標開始截取
NSString *key = [name substringFromIndex:1];
// 根據成員屬性名去字典中查找對應的value
id value = dict[key];
// 二級轉換:如果字典中還有字典,也需要把對應的字典轉換成模型
// 判斷下value是否是字典
if ([value isKindOfClass:[NSDictionary class]]) {
// 字典轉模型
// 獲取模型的類對象,調用modelWithDict
// 模型的類名已知,就是成員屬性的類型
// 獲取成員屬性類型
NSString *type = [NSString stringWithUTF8String:ivar_getTypeEncoding(ivar)];
// 生成的是這種@"@\"User\"" 類型 -》 @"User" 在OC字符串中 \" -> ",\是轉義的意思,不佔用字符
// 裁剪類型字符串
NSRange range = [type rangeOfString:@"\""];
type = [type substringFromIndex:range.location + range.length];
range = [type rangeOfString:@"\""];
// 裁剪到哪個角標,不包括當前角標
type = [type substringToIndex:range.location];
// 根據字符串類名生成類對象
Class modelClass = NSClassFromString(type);
if (modelClass) { // 有對應的模型才需要轉
// 把字典轉模型
value = [modelClass modelWithDict:value];
}
}
if (value) { // 有值,才需要給模型的屬性賦值
// 利用KVC給模型中的屬性賦值
[objc setValue:value forKey:key];
}
}
return objc;
}
@end