对模型对象进行归档(objective-c)
对模型对象进行归档是4种将数据持久存储在iOS文件系统的机制之一
使用注意:只要在类中实现的每个属性都是标量(枚举类型、字符型和各种整数表达形式)或都是遵循NSCoding协议的某个类的实例,就可以对整个对象进行完全的归档。大多数支持存储数据的Foundation和Cocoa Touch类都遵循NSCoding协议(不过有一些例外,如UIImage)。
尽管对归档没有严格要求,但一般NSCopying协议应该与NSCoding协议一起实现,这个协议允许复制对象,这在使用数据模型对象时具备较大的灵活性。
一些进行归档和归档重组前的方法
获取Document目录,为了将之后归档完的数据存放于此
-(NSString*)dataFilePath
{
NSArray *paths=NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
//传递了三个参数,分别代表的是正在查找的路径,此次搜索限制在应用沙盒内,是否展开完整路径(YES)
NSString *documentsDirectory=[paths objectAtIndex:0];
//每个应用只有一个Document路径,所以返回数组的第一个就是所需项
return [documentsDirectory stringByAppendingPathComponent:@"data.archiver"];
//在路径的结尾附上文件名就可以创建指向该文件的路径,如果不存在该文件,系统会自动创建一个文件。
}
归档和归档重组需要遵循的两个协议
NSCoding协议
NSCoding协议声明了两个方法,一个方法将对象编码到归档中,另外一个方法对归档解码来创建一个新对象。
-
将对象编码到归档中
- 有一个注意点,如果需要子类化某个遵循NSCoding的类,还需要对其父类调用encodeWithCoder方法
-(void)encodeWithCoder:(NSCoder*)encoder { [super encodeWithCoder:encoder];//符合上述注意点时才调用! [encoder encodexxx:xxx forkey:xxx]; ... }
-
恢复之前归档的对象
与归档方法类似,当为某个有父类而且遵循NSCoding的类实现NSCOding时,initWithCoder:方法稍有不同
挺拗口的,实际上想表达的意思是,你要让一个类遵循NSCoding协议的话,需要去实现他的两个requied方法,然后当你要实现这个initWithCoder:方法的时候要注意,你需要判断现在这个类有没有父类,这个父类是不是也遵循NSCoding协议
-(id)initWithCoder:(NSCoder *)aDecoder { self=[super init];//普通调用 self=[super initWithCoder:aDecoder;//符合上面描述时调用 if (self) { xxx=[aDexcoder decodeObjectForKey:xx]; } return self; }
NSCopying协议
遵循NSCopying对于任何数据模型对象来说都是非常好的事情,NSCopying有一个copyWithZone的方法可以用来复制对象
-
此处copyWithZone方法内容如下(需要注意的是,在精通iOS开发这本书的数据持久化归档这一块中,虽然有遵循这一个协议,但是应该是没有实现调用里面的方法的)
-(id)copyWithZone:(NSZone *)zone
{
Myclass *copy=[[[self class]allocWithZone:zone]init];
copy.foo=[self.foo copyWithZone:zone];
...
return copy;
}
```
对数据对象进行归档和取消归档
创建归档
从遵循NSCoding的一个或多个对象创建归档,首先创建一个NSMutableData实例,用于包含编码的数据,然后创建一个NSKeyedArchiver实例,用于将对象归档到此NSMutableData实例中。
NSData和NSMutableData类所创建出来的实例是用于存放二进制数的。而此处的NSKeyedArchiver所创建实例,可以理解成为一个压缩工具,它将原文件化成二进制数,便于储存在Data实例中。
NSString *filePath=[self dataFilePath];
NSMutableData *data=[[NSMutableData alloc]init];
//创建NSMutableData实例
NSKeyedArchiver *archiver=[[NSKeyedArchiver alloc]initForWritingWithMutableData:data];
//创建NSKeyedArchiver实例
[archiver encodeObject:myObject forKey:@"xx"]
//这一句话可以理解为将这个myObject对象进行归档,使用键值编码,其中这个xxx的键就对应这个对象的值。在调用这条语句以后,函数会跳转到遵循NSCoding协议的myObject的所属的类中的encoderWithCoder:方法
[archiver finishEncoding];
[data writeToFile:filePath atomically:YES];
整个流程可以理解为,首先创建data对象,然后以data作为写入路径创建archiver对象,archiver发出消息,以xx为键将myObject对象归档,然后函数跳转至myObject所属的类中,执行encoderWithCoder:方法,将myObject里面所包含的属性,对象再以xxx键进行归档,然后存放在data对象中,然后将data数据根据Document路径写入。
从归档重组对象
创建归档和从归档重组基本可以看为一个互逆的过程,步骤跟思路也相似
NSString *filePath=[self dataFilePath];
NSData *data=[NSData alloc]initWithContentsOfFile:filePath;
//从归档文件创建一个NSData实例
NSKeyedUnarchiver *unarchiver=[[NSKeyedUnarchiver alloc]initForReadingWithData:data];
//unarchiver相当于一个只服务于data的解压器,用于对数据解码
self.xxx=[unarchiver decodeObjectForKey:xxx;
//这里的key是相对于归档时候所用的key而言
[unarchiver finishDecoding];
小结
- 以上代码片都是参考精通iOS开发这本书所写出来的,可以在这本书的第13章第4小节参考全部分代码
- 在这一部分的知识学习过程中遇到了两个问题,第一个是NSCoding实现方法中的key跟NSKeyedUnarchiver/NSKeyedarchiver中的key有什么不一样,第二个是NSCoding和NSCopying实现的方法在何处被调用
- 针对第一个问题,最后得出的答案是,NSKeyedUnarchiver/NSKeyedarchiver的Key针对的是整个对象的归档/重组,而NSCoding方法中的Key针对的是这个对象里面所包含的属性以及对象的归档/重组。
- 第二个问题,NSCopying通过设置断点发现在该书的整段代码中应该是没有被调用,而NSCoding的两个方法都是在NSKeyedUnarchiver/NSKeyedarchiver所对应的实例对象调用其方法的时候,函数跳转至NSCoding的实现方法。