詳解Obejective-C中將JSON數(shù)據(jù)轉(zhuǎn)為模型的方法
在我們的日常開發(fā)中需要對(duì)加載的一些本地?cái)?shù)據(jù)例如plist、json等文件中的數(shù)據(jù)進(jìn)行模型轉(zhuǎn)化,而蘋果也為我們提供了一種非常方便的鍵值轉(zhuǎn)換方式KVC。然而KVC在某些情況下并不能保存數(shù)據(jù)的轉(zhuǎn)換成功,比如必須保證模型的屬性個(gè)數(shù)大于等于字典個(gè)數(shù),也要必須屬性名稱與字典的key相同等。所以這次我們假設(shè)下屬性名稱與字典中的key不一致的時(shí)候轉(zhuǎn)換方法。
首先我們還是先要嘗試下使用KVC的方式來解決這個(gè)問題
模型如下:
@property (nonatomic, strong) NSString *title;
@property (nonatomic, strong) NSString *name;
@property (nonatomic, strong) NSString *ID;
JSON數(shù)據(jù)如下:
{ "title" : "順平侯", "name" : "趙云", "id" : "sph" }, { "title" : "恒侯", "html" : "張飛", "id" : "hh" }, { "title" : "威侯", "html" : "馬超", "id" : "wh" }, { "title" : "剛侯", "html" : "黃忠", "id" : "gh" }, { "title" : "壽亭侯", "html" : "關(guān)羽", "id" : "sth" }
從上面的數(shù)據(jù)對(duì)比我們不難發(fā)現(xiàn),因?yàn)樵贠C中的id是關(guān)鍵字所有我們使用ID來替代,但是這樣的話就不能直接使用KVC,所以我們需要進(jìn)行相應(yīng)的處理來繼續(xù)使用我們的KVC轉(zhuǎn)換模型。代碼如下:
首先在模型.h文件中更新一下代碼,提供一個(gè)類方法進(jìn)行模型轉(zhuǎn)換:
@property (nonatomic, strong) NSString *title;
@property (nonatomic, strong) NSString *name;
@property (nonatomic, strong) NSString *ID;
+(instancetype) heroDict:(NSDictionary*) dict;
在.m文件中實(shí)現(xiàn)該方法
+ (instancetype)itemWithDict:(NSDictionary *)dict
{
HeroItem *hero = [[self alloc]init];
[item setValuesForKeysWithDictionary:dict];
return item;
}
程序走到這里就會(huì)去模型中遍歷字典當(dāng)中的所有的key。所以我們要修改的地方也就是這里去重寫KVC中的setValue forKey方法.代碼如下:
- (void)setValue:(id)value forKey:(NSString *)key{
//因?yàn)橐呀?jīng)知道要修改的key所以可以直接判定相等
if ([key isEqualToString:@"id"]) {
//進(jìn)行替換
[self setValue:value forKeyPath:@"ID"];
}else{
//拋回父類處理
[super setValue:value forKey:key];
}
}
程序修改到這里,基本就可以使用KVC的方法進(jìn)行轉(zhuǎn)換。但是如果我們的數(shù)據(jù)有很多不一致的情況呢?那么就讓我們一起來看下今天的重頭戲runtime來的轉(zhuǎn)換。
上面的例子的思路是通過遍歷字典當(dāng)中的key去模型中比對(duì),而我們這次試著遍歷模型,然后去字典中比對(duì)響應(yīng)的key
首先在我們的模型.m里導(dǎo)入我們需要的頭文件
#import <objc/runtime.h>
完成這一步在模型類中就可以使用runtime了,然后我們?cè)?m中創(chuàng)建一個(gè)轉(zhuǎn)換的類方法
+ (instancetype)objcWithDict:(NSDictionary *)dict updateDict:(NSDictionary *)updateDict{
}
在這個(gè)方法中我們需要做的是通過runtime來遍歷模型中的屬性,進(jìn)行屬性的比對(duì),如果模型中的屬性在字典中不存在,那么就會(huì)去updateDict中尋找,如果updateDict字典中存在的話就會(huì)進(jìn)行轉(zhuǎn)換。objctWithDict:方法更新如下:
(instancetype)objcWithDict:(NSDictionary *)dict updateDict:(NSDictionary *)updateDict{
id objc = [[self alloc] init];
// 遍歷模型中屬性
unsigned int count = 0;
Ivar *ivars = class_copyIvarList(self, &count);
for (int i = 0 ; i < count; i++) {
Ivar ivar = ivars[i];
// 屬性名稱
NSString *ivarName = @(ivar_getName(ivar));
ivarName = [ivarName substringFromIndex:1];
id value = dict[ivarName];
// 模型中屬性名對(duì)應(yīng)字典中的key
if (value == nil) {
if (updateDict) {
NSString *keyName = updateDict[ivarName];
value = dict[keyName];
}
}
[objc setValue:value forKeyPath:ivarName];
}
return objc;
}
到了這里轉(zhuǎn)換已經(jīng)完成,那么我們更新下heroDict:方法代碼:
+ (instancetype)itemWithDict:(NSDictionary *)dict{
//調(diào)用方法,updateDict中的數(shù)據(jù)即為需要替換的數(shù)據(jù)
HeroItem *item = [HeroItem objcWithDict:dict updateDict:@{@"ID":@"id"}];
return item;
}
到了這里runtime轉(zhuǎn)換的方法也完成了。對(duì)比兩個(gè)方法的話可能明顯會(huì)發(fā)現(xiàn)第一種方法會(huì)比較簡(jiǎn)單。但是如果是多個(gè)模型的話就需要大量的來重寫setValue:方法了,而第二種方法則可以封裝起來用以適用于各種模型。當(dāng)然了如果真的是大型項(xiàng)目的話還是比較推薦使用一些非常優(yōu)秀的第三方框架來處理模型,比如MJ大神的MJExtension使用起來簡(jiǎn)單方便,絕對(duì)是開發(fā)的上上之選了。
使用jastor
如果有jastor這個(gè)庫(kù),也會(huì)方便很多現(xiàn)在就基本的用法做個(gè)介紹。
假如我們有如下這么一個(gè)類
#import <Foundation/Foundation.h>
#import "Jastor.h"
@interface DeviceEntity : Jastor
@property (nonatomic,strong) NSNumber *isonline;
@property (nonatomic,strong) NSNumber *isopen;
@property (nonatomic,copy) NSString *brand;
@end
#import "DeviceEntity.h"
@implementation DeviceEntity
@synthesize isopen,isonline,brand;
@end
#import <Foundation/Foundation.h>
#import "Jastor.h"
#import "DeviceEntity.h"
@interface UserDevicesEntity : Jastor
@property (nonatomic,strong) NSNumber *closecount;
@property (nonatomic,strong) NSNumber *opencount;
@property (nonatomic,copy) NSString *success;
@property (nonatomic,strong) NSArray *items;
@end
#import "UserDevicesEntity.h"
#import "DeviceEntity.h"
@implementation UserDevicesEntity
@synthesize closecount,opencount,success,items;
+ (Class) items_class {
return [DeviceEntity class];
}
@end
注意這里在定義相應(yīng)屬性的時(shí)候如果是基本類型我們需要用NSNumber來進(jìn)行包裝,上面的例子也表明了我們可以用數(shù)組來做為一個(gè)屬性,只在是實(shí)現(xiàn)的時(shí)候需要告訴它這個(gè)數(shù)組是什么類型的,你定義的屬性名后跟_class的形式,注意這一點(diǎn)不能搞錯(cuò)。
在調(diào)用服務(wù)的時(shí)候,對(duì)方一般都會(huì)返回一個(gè)json,我們要做的就是根據(jù)這個(gè)字符串實(shí)例化一個(gè)NSDictionary出來,然后就可以根據(jù)這個(gè)NSDictionay實(shí)例化相應(yīng)的模型了,比我們直接解析這個(gè)字符串方便多了,代碼如下:
NSDictionary *dictionary = [NSDictionary dictionaryWithObjectsAndKeys:@"1",@"isonline",@"1",@"isopen",@"brand1",@"brand", nil];
DeviceEntity *device = [[DeviceEntity alloc] initWithDictionary:dictionary];
我們可以驗(yàn)證下,
NSLog(@"device's brand is %@",device.brand);
NSLog(@"device's isonline is %d",[device.isonline intValue]);
NSLog(@"device's isopen is %d",[device.isopen intValue]);
將會(huì)打印出
2014-02-17 22:36:37.602 objc-grammar-learing[819:f803] device's brand is brand1 2014-02-17 22:36:37.605 objc-grammar-learing[819:f803] device's isonline is 1 2014-02-17 22:36:37.605 objc-grammar-learing[819:f803] device's isopen is 1
看看是不是方便很多,當(dāng)然上面只是很簡(jiǎn)章的模型,一般來講,真實(shí)項(xiàng)目中的模型肯定比這復(fù)雜,比如一對(duì)一,一對(duì)多等等,在官網(wǎng)上面都有相應(yīng)例子可以參考。
相關(guān)文章
ios動(dòng)態(tài)設(shè)置lbl文字標(biāo)簽的高度
本文給大家分享的是ios動(dòng)態(tài)設(shè)置lbl文字標(biāo)簽的高度寬度的方法,一共給大家匯總了3種方法,小伙伴們根據(jù)自己的項(xiàng)目需求自由選擇。2015-05-05iOS拍照后圖片自動(dòng)旋轉(zhuǎn)90度的完美解決方法
今天開發(fā)一個(gè)拍照獲取照片的功能的時(shí)候, 發(fā)現(xiàn)上傳之后圖片會(huì)自動(dòng)旋轉(zhuǎn)90.在測(cè)試中發(fā)現(xiàn)只要是圖片大于2M, 系統(tǒng)就會(huì)自動(dòng)翻轉(zhuǎn)照片。下面小編通過本文給大家分享下解決辦法2016-12-12iOS中實(shí)現(xiàn)imageView任意角度旋轉(zhuǎn)的方法
這篇文章主要給大家介紹了關(guān)于iOS中實(shí)現(xiàn)imageView任意角度旋轉(zhuǎn)的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-12-12IOS應(yīng)用內(nèi)跳轉(zhuǎn)系統(tǒng)設(shè)置相關(guān)界面的方法
在iOS開發(fā)中,有時(shí)會(huì)有跳轉(zhuǎn)系統(tǒng)設(shè)置界面的需求,例如提示用戶打開藍(lán)牙或者WIFI,提醒用戶打開推送或者位置權(quán)限等,接下來通過本文給大家介紹IOS應(yīng)用內(nèi)跳轉(zhuǎn)系統(tǒng)設(shè)置相關(guān)界面的方法,喜歡的朋友參考下2016-02-02設(shè)計(jì)模式中的Memento備忘錄模式的在iOS App開發(fā)中的運(yùn)用
這篇文章主要介紹了設(shè)計(jì)模式中的Memento備忘錄模式的在iOS App開發(fā)中的運(yùn)用,Memento著重于捕獲和具體化當(dāng)前對(duì)象的內(nèi)部狀態(tài),需要的朋友可以參考下2016-03-03iOS11解決UITableView側(cè)滑刪除無(wú)限拉伸的方法
這篇文章主要給大家介紹了關(guān)于iOS11如何解決UITableView側(cè)滑刪除無(wú)限拉伸的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-08-08