经过这两天的研究考察,总共得出两种model数组归解档方法,现记录如下:
方法1:首先把modle对象转化成字典,把model数组转成对应的字典数组。然后把数组进行存档操作。
下面给出的是利用runtime抽取model对象的所有属性的方法,和把一个model转化成字典的方法。当然,读档之后需要再重新建立model数组。
缺点:代码较多,略繁琐,不太推荐。
#import <objc/runtime.h>
//获取对象的所有属性
NSArray
{
u_int
objc_property_t *properties =class_copyPropertyList([selfclass], &count);
NSMutableArray *propertiesArray = [NSMutableArrayarrayWithCapacity:count];
for (int i =0; i<count; i++)
{
const char* propertyName =property_getName(properties[i]);
addObject: [NSStringstringWithUTF8String: propertyName]];
}
free(properties);
return
}
到字典
NSDictionary
{
NSMutableDictionary *props = [NSMutableDictionarydictionary];
unsigned int
objc_property_t *properties = class_copyPropertyList([self class], &outCount);
for (i = 0; i<outCount; i++)
{
objc_property_t
const char* char_f =property_getName(property);
NSString *propertyName = [NSStringstringWithUTF8String:char_f];
id propertyValue = [selfvalueForKey:(NSString
if (propertyValue) [props setObject:propertyValue forKey:propertyName];
}
free(properties);
return props;
}
下面是解档归档具体操作:
需要注意的是需要把数组以复杂对象归档的方式进行归档(推荐plist文件)。
//创建缓存地址
NSString *paths = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
NSLog(@"Caches: %@", paths);
NSMutableArray *personsArr = [NSMutableArray arrayWithObjects:personA, personB, personC, nil];
NSString *personsArrPath = [paths stringByAppendingString:@"/personsArr.plist"];
//归档
[NSKeyedArchiver archiveRootObject:personsArr toFile:personsArrPath];
NSLog(@"personsArrPath: %@", personsArrPath);
//解档
NSArray *newPersonsArr = [NSKeyedUnarchiver unarchiveObjectWithFile:personsArrPath];
NSLog(@"反归档: %@", newPersonsArr);
结合runtime和kvc,更简洁的实现如下两个协议方法。当然,如果你需要缓存的model数组或类很多,则可以创建一个BaseModel,实现如下方法,其他model类继承此类,即可实现归解档操作。
较推荐。
- (void)encodeWithCoder:(NSCoder *)encoder
{
unsigned int count = 0;
// 利用runtime获取实例变量的列表
Ivar *ivars = class_copyIvarList([self class], &count);
for (int i = 0; i < count; i ++) {
// 取出i位置对应的实例变量
Ivar ivar = ivars[i];
// 查看实例变量的名字
const char *name = ivar_getName(ivar);
// C语言字符串转化为NSString
NSString *nameStr = [NSString stringWithCString:name encoding:NSUTF8StringEncoding];
// 利用KVC取出属性对应的值
id value = [self valueForKey:nameStr];
// 归档
[encoder encodeObject:value forKey:nameStr];
}
// 记住C语言中copy出来的要进行释放
free(ivars);
}
- (id)initWithCoder:(NSCoder *)decoder
{
if (self = [super init]) {
unsigned int count = 0;
Ivar *ivars = class_copyIvarList([self class], &count);
for (int i = 0; i < count; i ++) {
Ivar ivar = ivars[i];
const char *name = ivar_getName(ivar);
//
NSString *key = [NSString stringWithCString:name encoding:NSUTF8StringEncoding];
id value = [decoder decodeObjectForKey:key];
// 设置到成员变量身上
[self setValue:value forKey:key];
}
free(ivars);
}
return self;
}
关于nscode,再补充一点:
NScoder 和 NScoding 有将自己定义的类的对象写入磁盘的作用
NScoding 是一个协议,主要有下面两个方法
-(id)initWithCoder:(NSCoder *)coder;//从coder中读取数据,保存到相应的变量中,即反序列化数据
-(void)encodeWithCoder:(NSCoder *)coder;// 读取实例变量,并把这些数据写到coder中去。序列化数据
NSCoder 是一个抽象类,抽象类不能被实例话,只能提供一些想让子类继承的方法。
NSKeyedUnarchiver 从二进制流读取对象。
NSKeyedArchiver 把对象写到二进制流中去。
一个简单的例子如下:
一般是在自己定义的类中需要在.h 文件中加入<NScoding>
在.m 文件众实现他的的两个代理方法,这个代理方法将会被自动调用
- (void)encodeWithCoder:(NSCoder *)aCoder
{
[aCoder encodeObject:self.InsureSolutionID forKey:@"personName"];
[aCoder encodeObject:self.InsureSolutionName forKey:@"personAge"];
}
encodeWithCoder 可以调用的方法:
1)、如果是类 就用encodeObject: forKey:
2)、如果是普通的数据类型就用 eg、encodeInt: forKey:
- (id)initWithCoder:(NSCoder *)aDecoder
{
self = [super init];
if (self)
{
self.InsureSolutionID = [aDecoder decodeObjectForKey:@"personName"];
self.InsureSolutionName = [aDecoder decodeObjectForKey:@"personAge"];
}
return self;
}
initWithCoder 可以调用的方法:
1)、如果是类 就用decodeObjectForKey:
2)、如果是普通的数据类型就用 eg、decodeIntForKey:
以下是对该类序列化和反序列化。
NSData *archiveCarPriceData = [NSKeyedArchiver archivedDataWithRootObject:self.DataArray];
[[NSUserDefaults standardUserDefaults] setObject:archiveCarPriceData forKey:@"DataArray"];
NSData *myEncodedObject = [[NSUserDefaults standardUserDefaults] objectForKey:@"DataArray"];
self.dataList = [NSKeyedUnarchiver unarchiveObjectWithData: myEncodedObject];
当然,如果你有更好的方法,欢迎补充。