iOS-字典转双模型的实现过程中需要关注的细节
如果有以上结构的plist文件,那么应该怎么将其中的字典转换成模型?
显然一个模型已经无法搞定了,此时需要用到双数据模型(字典转模型)。
我写了两种方式来实现模型的转换:
方法一
第一个模型:CarModel
两个属性:
@property (nonatomic,copy)NSArray *cars;
@property (nonatomic,copy)NSString *title;
第二个模型:InnerCarModel
三个属性:
@property (nonatomic,copy)NSString *icon;
@property (nonatomic,copy)NSString *name;
@property (nonatomic,strong)CarModel *carModel;
首先看carDateModel.h
#import <UIKit/UIKit.h>
@interface CarModel : UIView
@property (nonatomic,copy) NSArray *cars;
@property (nonatomic,copy) NSString *title;
-(instancetype)initWithCarDict:(NSDictionary*)dict;
+(instancetype)carModel:(NSDictionary*)dict;
@end
这个carDateModel中有两个属性
为什么要提供类方法和对象方法?
在controller的数据加载中;
@property(nonatomic, stong)NsmutableArray*arrayDate;
在使用该数组时,我们是这样赋值的 eg: xxxx = self.arrayDate[i]
这种方式赋值会调用arrayDate 的getter方法,所以我们需要重写arrayDate的getter方法;
#import "ViewController.h"
#import "InnerCarModel.h"
#import "CarModel.h"
@interface ViewController () <UITableViewDataSource,UITableViewDelegate>
@property (weak, nonatomic) IBOutlet UITableView *tableView;
@property (nonatomic,strong) NSMutableArray *arrayDate;
@property (nonatomic,strong) NSMutableArray *indexDate;
@end
@implementation ViewController
-(NSMutableArray *)arrayDate{
if (nil == _arrayDate) {
_arrayDate = [NSMutableArray array];
// _indexDate = [NSMutableArray array];
NSString *path = [[NSBundle mainBundle] pathForResource:@"cars_total.plist" ofType:nil];
NSArray *tempArray = [[NSArray alloc] initWithContentsOfFile:path];
for (NSDictionary *dict in tempArray) {
CarModel *model = [[CarModel alloc] initWithCarDict:dict];
//here do something innercarmodel at carModel.m
InnerCarModel *innerCarModel = [[InnerCarModel alloc] init];
innerCarModel.carModel = model;
[_arrayDate addObject:model];
}
// 使用的kvc 找到数组中对应的属性, 返回一个数组
// 写到这个地方, 只加载一次
_indexDate = [_arrayDate valueForKeyPath:@"title"];
}
return _arrayDate;
}
我们需要明白这三句的意义:
1. CarDateModel *carDateModel = [[CarDateModelalloc]
initWithCarDict:dict]
此句使用了我们在carDateModel.h中为外界提供的方法将数据放到carDateModel中;
2.InnerCarModel *innerModel = [[InnerCarModel alloc] init];
此句我们实例化了一个innerModel,
InnerCarModel怎么定义的?
#import
<UIKit/UIKit.h>
@class
CarModel;
@interface InnerCarModel :
UIView
@property (nonatomic,copy)NSString *icon;
@property (nonatomic,copy)NSString *name;
@property (nonatomic,strong)CarModel *carModel;
@end
是的,没错,在InnerCarModel Model中包含了一个carModel属性;
3. InnerModel.carDatemodel =carDateModel;
“=”右侧:carDateModel就是数据,数据中国包含了cars 和 title,在此我们需要的书cars。
“=”左侧InnerModel.carDatemodel,该句则是调用了carModel的setter方法。
所以我们需要重写carModel的setter方便我们对InnerCarModel的属性icon和name赋值
#import "InnerCarModel.h"
#import "CarModel.h"
@implementation InnerCarModel
-(void)setCarModel:(CarModel *)carModel {
_carModel = carModel;
NSMutableArray *mutbleArray = [NSMutableArray array];
for (NSDictionary *dict in carModel.cars) {
InnerCarModel *innerModel = [[InnerCarModel alloc] initWithInnerCarDict:dict];
[mutbleArray addObject:innerModel];
}
_carModel.cars = mutbleArray;
}
-(instancetype)initWithInnerCarDict:(NSDictionary*)dict{
if (self = [super init]) {
[self setValuesForKeysWithDictionary:dict];
}
return self;
}
+(instancetype)innerCarModel:(NSDictionary*)dict{
return [[self alloc]initWithInnerCarDict:dict];
}
@end
initWithInnerCarDict innerCarModel
这两个方法是私有的,不必在.h中声明。
以上,我们就完成了双模型的转换。
方法二
第一个模型
两个属性:
@property (nonatomic,copy) NSArray *cars;
@property (nonatomic,copy) NSString *title;
第二个模型:
两个属性:@property(nonatomic,copy) NSString *icon;
@property (nonatomic,copy) NSString *name;
不同之处:
加载数据的方式
</pre><pre name="code" class="objc">-(NSMutableArray *)arrayDate{
if (nil == _arrayDate) {
_arrayDate = [NSMutableArray array];
// _indexDate = [NSMutableArray array];
NSString *path = [[NSBundle mainBundle] pathForResource:@"cars_total.plist" ofType:nil];
NSArray *tempArray = [[NSArray alloc] initWithContentsOfFile:path];
for (NSDictionary *dict in tempArray) {
CarModel *model = [[CarModel alloc] initWithCarDict:dict];
[_arrayDate addObject:model];
}
_indexDate = [_arrayDate valueForKeyPath:@"title"];
}
return _arrayDate;
}
关键句1 CarModel *model = [[CarModel alloc] initWithCarDict:dict];
进入initWithCarDict:方法
</pre><pre name="code" class="objc">-(instancetype)initWithCarDict:(NSDictionary*)dict{</span></span>
if (self = [super init]) {
[self setValuesForKeysWithDictionary:dict];
NSMutableArray *mutbleArray = [NSMutableArray array];
for (NSDictionary *dict in self.cars) {
InnerCarModel *innerModel = [[InnerCarModel alloc] initWithInnerCarDict:dict];
[mutbleArray addObject:innerModel];
}
self.cars = mutbleArray;
}
return self;
}
此时,此中方式是,直接调用了initWithInnerCarDict方法,加载了self.cars 和title为 CarModel,
再将的self.cars数据转化为 InnerCarModel ,再返回数据到CarModel *model ,
_arrayDate最终装载的就是转化好的数据,供我们去调用。
总之,字典转模型,重点要关注数据的存和取的过程,在这个过程中不能将: xxxx = self.arrayDate[i]
这么一句话简单的理解为赋值语句,你要想到这句话中,包含的getter方法,在这个getter方法中,可能包含着setter 方法;
我们费了这么劲儿去转模型,就是为了:
1)降低代码的耦合度
2)所有字典转模型部分的代码统一集中在一处处理,降低代码出错的几率
3)在程序中直接使用模型的属性操作,提高编码效率
4)调用方不用关心模型内部的任何处理细节
需要根据外部的需求,提供相应的接口方法。
以上程序用UItableView实现了车品牌的展示,并建立相关的索引
程序源码在:
方法1:http://download.csdn.net/detail/yang198907/9222089
方法2: http://download.csdn.net/detail/yang198907/9222091
iOS-字典转双模型的实现过程中需要关注的细节的更多相关文章
- iOS 字典自动生成模型
在实际开发中,我们经常需要根据字典来建模型.每次都打那么一串代码,想想也是挺恶心的.可以自己给NSDictionary写一个分类,进行属性生成. NSDictionary+Property.h #im ...
- iOS:使用block代码块实现事件处理过程中的回调
block是什么,这里就不多加强调了,它的优点: 第一:执行效率高,速度快 第二:使用起来比代理简单,省却不少代码,增强代码美感 有一些小的知识点要强调一下: 第一点:它类似于一个匿名函数,也跟jav ...
- Web CI过程中的Security解决方案
http://www.infoq.com/cn/articles/WebScan-CI 一. 当前Web应用安全现状 随着中国互联网金融的爆发和繁荣,Web应用在其中扮演的地位也越来越重要,比如Web ...
- IOS中将字典转成模型对象
作为IOS开发初级者今天学习了 如何将plist数据字典转成 数据对象数组中 .有点像C#中解析xml数据 的过程. apps.plist的xml数据是这样的 <?xml version=&qu ...
- IOS 字典快速转换为Model 模型
一般情况下IOS得局部页面加载的过程是,创建一个Model然后,将Nib文件与Model进行关联,然后能够快速的获取到Nib文件上的控件实例.操作生成页面. 但是原生的内容是没有直接通过Json获取M ...
- IOS 字典模型互转框架 MJExtension
IOS 字典模型互转框架 MJExtension 能做什么? MJExtension是一套字典和模型之间互相转换的超轻量级框架 MJExtension能完成的功能 字典(JSON) --> ...
- iOS字典转模型MJExtension使用
如果项目是纯OC的建议使用,MJExtension是一套字典和模型之间互相转换的超轻量级框架,可以轻松完成: 字典(JSON) --> 模型(Model) 模型(Model) --> 字典 ...
- iOS 字典或者数组和JSON串的转换
在和服务器交互过程中,会iOS 字典或者数组和JSON串的转换,具体互换如下: // 将字典或者数组转化为JSON串 + (NSData *)toJSONData:(id)theData { NSEr ...
- OSI七层协议模型及OSI参考模型中的数据封装过程
转载自:http://blog.csdn.net/qq_14935437/article/details/71081546 OSI模型,即开放式通信系统互联参考模型(Open System Inter ...
随机推荐
- .Net Core3.0 WebApi 项目框架搭建:目录
一.目录 .Net Core3.0 WebApi 项目框架搭建 一:实现简单的Resful Api .Net Core3.0 WebApi 项目框架搭建 二:API 文档神器 Swagger .Net ...
- Java实现DDD中UnitOfWork
Java实现DDD中UnitOfWork 背景 Maintains a list of objects affected by a business transaction and coordinat ...
- Python的逻辑结构和函数
1.Python中的逻辑结构 ①顺序执行 ②选择执行: if...elif...else... 没有switch..case.. ③循环执行: for...in... while... 没有do..w ...
- 7.2 Go type assertion
7.2 Go type assertion 类型断言是使用在接口值上的操作. 语法x.(T)被称为类型断言,x代表接口的类型,T代表一个类型检查. 类型断言检查它操作对象的动态类型是否和断言类型匹配. ...
- 02.drf不使用serializers返回数据
drf 可以使用不经过model和serialzier的数据返回,也可以配置权限 class DashboardStatusViewset(viewsets.ViewSet): "" ...
- 如何分析和提高(C/C++)程序的编译速度?
版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://www.cnblogs.com/lihuidashen/p/129354 ...
- LOL源代码娜美皮肤免费领取
领取地址 http://t.cn/EyOY8zp 截图
- 聊聊UDP、TCP和实现一个简单的JAVA UDP小Demo
最近真的比较忙,很久就想写了,可是一直苦于写点什么,今天脑袋灵光一闪,觉得自己再UDP方面还有些不了解的地方,所以要给自己扫盲. 好了,咱们进入今天的主题,先列一下提纲: 1. UDP是什么,UDP适 ...
- Linux操作系统分析 | 深入理解系统调用
实验要求 1.找一个系统调用,系统调用号为学号最后2位相同的系统调用 2.通过汇编指令触发该系统调用 3.通过gdb跟踪该系统调用的内核处理过程 4.重点阅读分析系统调用入口的保存现场.恢复现场和系统 ...
- C# 7.0 新增功能&结合微软简化理解
C# 7.0更新时间为2019.2左右 C# 7.0 ~ 7.3 分别需要VS2017 与 .NET Core 1.0. .NET Core 2.0 SDK..NET Core 2.1 SDK,需要在 ...