文章目录
什么是JSONModel?
-
做移动端开发,解析网络数据是必不可少的工作之一。iOS原生框架很早前就已经提供了将JSON数据直接映射成数组或者字典对象的方法,并且结合KVC,也可以将字典数据直接赋值给对象。但是这种方式十分不灵活,例如如果网络数据中的字段与我们数据模型中的字段不一致,某些网络数据的字段可能为
nil
等等都需要开发者单独的处理。使用
JSOMModel
可以十分方便的处理映射过程中的各种情况。
越来越多的移动应用更倾向于用JSON这种数据格式。一旦计划开发移动应用的并有与后台通信的需求,则要用JSON数据格式与服务器通信互相通信
-
JSONModel
是用
Objective-C
写的开源库。它包含了
接受发送
、
解析JSON数据
,用JSON数据
驱动初始化你的类
,还能够
检验JSON和嵌套模型
等功能。
核心数据模型类JSONModel
-
平时在使用
JSOMModel
框架时,往往只会用到
JSOMModel
这一个类。
JSONModel
框架中最核心的类是
JSONModel
类,其中代码大约有1400行。首先,其头文件中声明了几个协议,如下:
@protocol Index
@end
@protocol Ignore
@end
@protocol Optional
@end
-
需要注意,这些协议里面都没有约定任何方法,它们也不会用来实现的,其作为属性的一种标记,例如将属性添加
Ignore
协议,则
JSONModel
不会对这个属性进行解析、使用这种方式来进行本地数据的管理,例如:
@interface MyOnject : JSONModel
@property(nonatomic, strong) NSString * firstName;
@property(nonatomic, strong) NSString * lastName;
//这个属性是本地拼接 使用
@property(nonatomic, strong) NSString<Ignore> * fullName;
@end
-
Optional
协议表示这个属性是可选的,即JSON数据中如果有这个属性就解析,如果没有就跳过。
a. 某些属性值可以为空;
b. 防止由于服务器返回数据为空导致
JSONModel
异常(程序崩溃); -
Index
协议标记这个属性是当前对象的主键,
已经弃用
。 -
有了这3个协议,在声明属性时,我们可以十分容易的设定他们的解析规则,在
JSONModel
中,协议除了可以用来规定解析规则外,还可以用来指定自定义数据类型的解析,只是我们需要自己定义一个协议,名称与自定义类名一致,示例如下:
#import "JSONModel.h"
@protocol Address
@end
@interface Address:JSONModel
@property(nonatomic, strong) NSString * info;
@end
@interface MyObject : JSONModel
@property(nonatomic, strong) NSArray<Address> * address;
@end
-
如上代码所示,在解析数据时,会直接将
address
数组中赋值为Address的对象
JSON的一些基本使用
JSON转化为模型
//JSON
{ "country": "Germany",
"dialCode": 49,
"isInEurope": true
}
//Model
#import "JSONModel.h"
NS_ASSUME_NONNULL_BEGIN
@interface CountryModel : JSONModel
@property (nonatomic, copy) NSString *country;
@property (nonatomic, copy) NSString *dialCode;
@property (nonatomic, assign) BOOL isInEurope;
@end
NS_ASSUME_NONNULL_END
//JSON转换为模型
NSError *error;
CountryModel *countryModel = [[CountryModel alloc] initWithDictionary:dic error:&error];
模型转换为字典
//将模型快速转换为字典
NSDictionary *dict = [countryModel toDictionary];
模型转换为字符串
//将模型快速转换为字符串
NSString *string = [countryModel toJSONString];
设置所有属性可选(所有属性值可以为空)
a.
Model
的所有属性值可以为空;
b. 防止由于服务器数据返回空导致
JSONModel
异常(程序崩溃);
c. 官方建议
尽量避免
使用该方法(即使要全部属性为可选,也尽量是在每个属性那里标注为
Optional
);
eg:
@implementation TestModel
+ (BOOL)propertyIsOptional:(NSString *)propertyName {
return YES;
}
@end
对于集合、嵌套型数据
例如下面:
{
"date":"20211011",
"stories":[
{
"image_hue":"0x555555",
"title":"出现低落焦虑时,怎么分辨是「短暂情绪」还是生病了?",
"url":"https:\/\/daily.zhihu.com\/story\/9741137",
"hint":"精神科医生宋崇升 · 4 分钟阅读",
"ga_prefix":"101107",
"images":[
"https:\/\/pic3.zhimg.com\/v2-e68e99fafc8f87cd3c91a17a6f835d1f.jpg?source=8673f162"
],
"type":0,
"id":9741137
},
Object{...},
Object{...},
Object{...},
Object{...},
Object{...}
],
"top_stories":[
{
"image_hue":"0xb38d51",
"hint":"作者 \/ 男爵兔",
"url":"https:\/\/daily.zhihu.com\/story\/9741077",
"image":"https:\/\/pica.zhimg.com\/v2-ca0fe2412807776234bf0019f7276541.jpg?source=8673f162",
"title":"2021 年诺贝尔文学奖授予小说家阿卜杜勒拉扎克·古尔纳,他是谁?",
"ga_prefix":"100807",
"type":0,
"id":9741077
},
Object{...},
Object{...},
Object{...},
Object{...}
]
}
- 这次的数据很复杂他有嵌套, 有数组,我们应该怎样处理这种嵌套模型呢?我们应该对每一个要嵌套的都写成一个类,但并不是意味着要写成多个类文件,而是只需要在一个类文件里把该有的写好就行 如下代码:
@protocol StoriesModel
@end
@protocol Top_StoriesModel
@end
#import "JSONModel.h"
NS_ASSUME_NONNULL_BEGIN
@interface StoriesModel : JSONModel
@property (nonatomic, copy) NSString *title;
@property (nonatomic, copy) NSString *ga_prefix;
@property (nonatomic, copy) NSString *type;
@property (nonatomic, copy) NSString *image_hue;
@property (nonatomic, copy) NSString *id;
@end
@interface Top_StoriesModel : JSONModel
@property (nonatomic, copy) NSString *title;
@property (nonatomic, copy) NSString *ga_prefix;
@property (nonatomic, copy) NSString *type;
@property (nonatomic, copy) NSString *image_hue;
@property (nonatomic, copy) NSString *id;
@end
@interface TestModel : JSONModel
@property (nonatomic, copy) NSString *date;
@property (nonatomic, copy) NSArray<StoriesModel>* stories;
@property (nonatomic, copy) NSArray<Top_StoriesModel>* top_stories;
@end
NS_ASSUME_NONNULL_END
设置下划线自动转驼峰
-
自定义把下划线字段解析为
驼峰命名
属性 -
场景:服务器数据返回下划线命名字段可为
Model
中以驼峰命名的属性相应的赋值 -
mapperFromUpperCaseToLowerCase
大写转小写
{
"order_id": 104,
"order_product" : @"Product#1",
"order_price" : 12.95
}
@interface OrderModel : BaseModel
@property (nonatomic, strong) NSString *orderId;
@property (nonatomic, assign) float orderPrice;
@property (nonatomic, strong) NSString *orderProduct;
@end
@implementation OrderModel
+ (JSONKeyMapper *)keyMapper {
return [JSONKeyMapper mapperFromUnderscoreCaseToCamelCase];
}
@end
实例解析
-
首先要导入
JSONModel
库,与
Masonry
库的导入相同,修改库名即可 -
创建一个
TestModel
类,该类继承于
JSONModel
-
将请求到的
json
的数据在
.h
文件中声明为属性。因为涉及到数据的嵌套,因此用到上面所用到的方法,对每一个要嵌套的都写成一个类,把写好的类放在该类文件里
@protocol StoriesModel
@end
@protocol Top_StoriesModel
@end
#import "JSONModel.h"
NS_ASSUME_NONNULL_BEGIN
@interface StoriesModel : JSONModel
@property (nonatomic, copy) NSString *title;
@property (nonatomic, copy) NSString *ga_prefix;
@property (nonatomic, copy) NSString *type;
@property (nonatomic, copy) NSString *image_hue;
@property (nonatomic, copy) NSString *id;
@end
@interface Top_StoriesModel : JSONModel
@property (nonatomic, copy) NSString *title;
@property (nonatomic, copy) NSString *ga_prefix;
@property (nonatomic, copy) NSString *type;
@property (nonatomic, copy) NSString *image_hue;
@property (nonatomic, copy) NSString *id;
@end
@interface TestModel : JSONModel
@property (nonatomic, copy) NSString *date;
@property (nonatomic, copy) NSArray<StoriesModel> *stories;
@property (nonatomic, copy) NSArray<Top_StoriesModel> *top_stories;
@end
NS_ASSUME_NONNULL_END
在
TestModel.m
文件中一般不需要做其他事,但是为了防止由于服务器数据
返回空
导致
JSONModel
异常(程序崩溃),所以在
.m
中添加如下方法函数
#import "TestModel.h"
@implementation Top_StoriesModel
+ (BOOL)propertyIsOptional:(NSString *)propertyName {
return YES;
}
@end
@implementation StoriesModel
+ (BOOL)propertyIsOptional:(NSString *)propertyName {
return YES;
}
@end
@implementation TestModel
+ (BOOL)propertyIsOptional:(NSString *)propertyName {
return YES;
}
@end
-
此时解析的流程基本完成,剩下就是将数据通过网络请求传过来,并将
model
初始化,如下
- (void)viewDidLoad {
[super viewDidLoad];
NSString *json = @"http://news-at.zhihu.com/api/4/news/latest";
json = [json stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLQueryAllowedCharacterSet]];
NSURL *testUrl = [NSURL URLWithString:json];
NSURLRequest *testRequest = [NSURLRequest requestWithURL:testUrl];
NSURLSession *testSession = [NSURLSession sharedSession];
NSURLSessionDataTask *testDataTask = [testSession dataTaskWithRequest:testRequest completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
TestModel *country = [[TestModel alloc] initWithData:data error:nil];
NSLog(@"%@",country.stories[0]);
}];
//任务启动
[testDataTask resume];
}
-
如果传过来的
JSON
合法,你所定义的所有的属性都会与该
JSON
值相匹配,并且
JSONModel
也会尝试尽可能的转换成你所想要的数据
此时输出如下:
遇到的问题
- 写完如上代码后网络请求时会报错,如下
-
原因如下:
iOS9引入了新特性:App Transport Security (ATS),新特性要求App内访问的网络必须使用HTTPS协议。
但是现在公司的项目使用的是HTTP协议,使用私有加密方式保证数据安全。现在也不能马上改成HTTPS协议传输。
-
解决方案:
在
Info.plist
中添加如下后即可正常进行网络请求了
-
如果我想在
viewDidLoad
中获取嵌套数组中的元素,例如:
NSLog(@"%@",country.stories[0].title);
-
此时不能直接进行打印,因为在该文件中我们没有事先声明,此处是
model
嵌套
model
,所以声明一下即可
@property (nonatomic, copy) StoriesModel *stories;
-
将访问到的
country.stories[0]
赋给刚才声明的属性,
因为是在block中进行的操作,所以不能直接使用
_stories = country.stories[0];
-
在代码中使用
_stories
时,编译器将用
self->_stories
替换代码,并且如果在块内使用它,则该块将捕获
self
自身而不是
stories
本身。 警告只是为了确保开发人员了解此行为。
self->_stories = country.stories[0];
NSLog(@"%@",self->_stories.title);
版权声明:本文为kochunk1t原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。