一、简单介绍setValuesForKeysWithDictionary
在iOS的日常开发中,经常用到model来明确所使用的数据模型,这样子可视化、数据逻辑等都明显高于字典。之前公司旧项目没有使用model来模型化字典,所有的数据都是在VC中用字典来存储(一个人负责的,估计是为了省事。。。可实际上感觉费了更多的代码和时间,同时这个是一个静态库项目,我的天,要看一个数据模型就要跑项目打印数据,到我这里就费劲)。
所以,将数据模型化,无论在MVC、MVVM还是我公司最近用的MVP架构中都是十分必要的,不仅仅提供方便了其他人快速了解字典内的数据,同时也可以将数据与页面分离。
在字典模型化下,setValuesForKeysWithDictionary简直不要太帅,没有使用其的情况下,我们对于字典模型化,是一条条的赋值,比如:
NSMutableDictionary * dic = [NSMutableDictionary new];
[dic setObject:@"a" forKey:@"name"];
[dic setObject:@"mail" forKey:@"sex"];
AModel *model = [AModel new];
model.name = dic[@"name"];
model.sex = dic[@"sex"];
这个样子,假如数据量大的话,不仅代码量大,为了一个这么没有技术含量的工作浪费大量的时间也是没有必要的。所以,苹果提供了setValuesForKeysWithDictionary来解决这种问题。上面的例子,可以这样来实现。
NSMutableDictionary * dic = [NSMutableDictionary new];
[dic setObject:@"a" forKey:@"name"];
[dic setObject:@"mail" forKey:@"sex"];
AModel * model = [AModel new];
[model setValuesForKeysWithDictionary:dic];
因为在项目中使用,有多个model的情况,可以创建一个基类BaseModel来管理,所有的继承这个类,将setValuesForKeysWithDictionary放入init方法中。
#import "BaseModel.h"
@implementation BaseModel
// kvc 赋值
- (instancetype)initValueWithDictionary:(NSDictionary *)dic{
self = [super init];
if (self) {
[self setValuesForKeysWithDictionary:dic];
}
return self;
}
@end
setValuesForKeysWithDictionary非常好用,不需要你来一一的给对象赋值而直接从字典初始化即可,注意的是模型与字典的key值必须对应。用的不好会经常崩溃!我们可以来看一下两种情况:
1、字典中无而model中有的数据
//model 有name sex mobile
NSMutableDictionary * dic = [NSMutableDictionary new];
[dic setObject:@"a" forKey:@"name"];
[dic setObject:@"mail" forKey:@"sex"];
AModel *model = [[AModel alloc] initValueWithDictionary:dic];
正常运行,model中的mobile打印出来是 null。
2、字典中有model中没有的数据
//model 只有name sex
NSMutableDictionary * dic = [NSMutableDictionary new];
[dic setObject:@"a" forKey:@"name"];
[dic setObject:@"mail" forKey:@"sex"];
[dic setObject:@"厦门" forKey:@"address"];
AModel *model = [[AModel alloc] initValueWithDictionary:dic];
这种情况会崩溃,错误
*** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: '[<AModel 0x17403e280> setValue:forUndefinedKey:]: this class is not key value coding-compliant for the key address.'
这种情况是因为,setValuesForKeysWithDictionary在模型中找不到对应的key--address 来进行赋值处理,所以出错崩溃。
这种情况其实很好解决,利用:
- (void)setValue:(id)value forUndefinedKey:(NSString *)key;
在model.m中实现这个方法,处理key值不存在或者是没有对应的情况。
//不做任何处理,即key值不存在的情况下
-(void)setValue:(id)value forUndefinedKey:(NSString *)key{
}
//model.m中
//key值没有对应的情况下
-(void)setValue:(id)value forUndefinedKey:(NSString *)key{
if ([key isEqualToString:@"UserName"]) {
self.name = value;
}
}
//VC中的赋值
//model 有name sex mobile
NSMutableDictionary * dic = [NSMutableDictionary new];
[dic setObject:@"a" forKey:@"UserName"];
[dic setObject:@"mail" forKey:@"sex"];
[dic setObject:@"厦门" forKey:@"address"];
AModel *model = [[AModel alloc] initValueWithDictionary:dic];
这样子就不会出现了崩溃的现象。
二、potocol管理子类的特殊情况。
在项目中model可以设立一个基类,所有的model都继承这个类,就像我上面所说的,因为大部分的model都有这两种行为:初始化和特殊key值处理。在基类与子类的关系中,我喜欢用potocol来管理差异(MVP)(个人理解:相比于子类重写基类的方法,应该是层次更清楚,让其他人直接明白子类需要的实现的协议)。
BaseModel.h
#import <Foundation/Foundation.h>
@protocol ModelProtocol <NSObject>
@optional
- (instancetype)initValueWithDictionary:(NSDictionary *)dic;
- (void)initValue:(id)value forUndefinedKey:(NSString *)key;
@end
@interface BaseModel : NSObject<ModelProtocol>
@end
BaseModel.m
#import "BaseModel.h"
@implementation BaseModel
// kvc 赋值
- (instancetype)initValueWithDictionary:(NSDictionary *)dic{
self = [super init];
if (self) {
[self setValuesForKeysWithDictionary:dic];
}
return self;
}
//没有找到模型key
- (void)setValue:(id)value forUndefinedKey:(NSString *)key{
if ([self respondsToSelector:@selector(initValue:forUndefinedKey:)]) {
[self initValue:value forUndefinedKey:key];
}
}
在子类中,实现特殊key值
- (void)initValue:(id)value forUndefinedKey:(NSString *)key{
if ([key isEqualToString:@"results"]) {
}
}
三、子类模型属性是一个类型为其他模型的数组。
model中有数组,数组内类型也是模型的话,也可以利用:
- (void)setValue:(id)value forUndefinedKey:(NSString *)key;
来实现。
AModel.h
#import "BaseModel.h"
@class BModel;
@interface AModel : BaseModel
@property (nonatomic, strong) NSString * name;
@property (nonatomic, strong) NSString * sex;
@property (nonatomic, strong) NSString * mobile;
@property (nonatomic, strong) NSMutableArray<BModel *>* bArray;
@end
AModel.m
#import "AModel.h"
#import "BModel.h"
@implementation AModel
- (void)initValue:(id)value forUndefinedKey:(NSString *)key{
if ([key isEqualToString:@"array"]) {
if ([value isKindOfClass:[NSArray class]]) {
for (NSDictionary *dic in value) {
[self.bArray addObject:[[BModel alloc] initValueWithDictionary:dic]];
}
}
}
}
//懒加载
-(NSMutableArray<BModel *> *)bArray{
if (_bArray == nil) {
_bArray = [NSMutableArray<BModel *> new];
}
return _bArray;
}
@end