1、什么叫归档
归档:即序列化。任何对象都可以遵循<NSCoding>协议进行归档。通过对数据模型对象进行归档可以轻松将复杂的对象写入文件,然后再从中读取它们。
另外,尽管对归档的使用没有严格要求,但还有一个协议应该与<NSCoding>一起实现,那就是<NSCopying>协议。后者允许复制对象,这会让你在使用数据模型对象时具备了较大的灵活性。
@protocol NSCoding
- (void)encodeWithCoder:(NSCoder *)aCoder;
- (id)initWithCoder:(NSCoder *)aDecoder; // NS_DESIGNATED_INITIALIZER
@end
2、如何归档
假设B为A的子类,A、B都遵循<NSCoding>协议,那么B类在实现<NSCoding>协议时,需要调用父类的对应方法,如果A没有遵循<NSCoding>协议那么B类在实现<NSCoding>协议时就不需要调用父类的对应方法了。如下:
- (void)encodeWithCoder:(NSCoder *)aCoder
{
[super encodeWithCoder:aCoder];//先调用父类的
[aCoder encodeObject:self.finishedLines forKey:@"finishedLines"];
}
- (id)initWithCoder:(NSCoder *)aDecoder
{
self = [super initWithCoder:aDecoder];
if (self)
{
self.finishedLines = [aDecoder decodeObjectForKey:@"finishedLines"];
}
return self;
}
////////////
- (void)encodeWithCoder:(NSCoder *)aCoder
{
// [super encodeWithCoder:aCoder];这里没调用super,是因为NSObject不遵循NSCoding协议,因此调不了,而BNRDrawView可以调super是因为UIView遵循NSCoding协议,系统应该实现了NSCoding协议的方法,所以可以调用super.
[aCoder encodeObject:[NSValue valueWithCGPoint:self.startPoint] forKey:@"startPoint"];
[aCoder encodeObject:[NSValue valueWithCGPoint:self.endPoint] forKey:@"endPoint"];
}
只要实现了这两个方法,就可以对所有对象的属性进行编码和解码,然后便可以对对象进行归档,并且可以将其写入归档或者从归档中读取它们。
xq注:把一个对象的所有属性进行编码和解码后,这个对象就可以进行归档了,但如果这个对象的属性的类型又是一个自定义类,那么该自定义类也需要把自身所有属性进行编码和解码。如果对象的属性的类型是一个容器类比如NSArray型的,如果这个容器装的是自定义类的对象,那么这个自定义类同样也需要把自身所有属性进行编码和解码。
3、归档后如何使用
示例:
归档方法①:
NSMutableData *data = [[NSMutableData alloc] init];
NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
[archiver encodeObject:drawView forKey:@"drawView"];//由第5点归档原理可知,drawView的- (void)encodeWithCoder:(NSCoder *)aCoder方法的参数aCoder就是NSKeyedArchiver对象archiver(这里用NSCoder类型,父类的指针可以指向子类的对象).
[archiver finishEncoding]; //至此对象序列化完成.
if ([data writeToFile:kSavePath atomically:YES])
另一种方便的归档使用方法②:
[NSKeyedArchiver archiveRootObject:self.privateItems toFile:path];
这种方法和上面那种方法原理上是一样的,方法②应该是苹果提供的一种简便方法.
工作原理如下:
- 这个方法首先创建了一个NSKeyedArchiver实例.(NSKeyedArchiver类是抽象类NSCoder的一个子类,是一个具体类)
- self.privateItems被发送消息encodeWithCoder:,并且被作为一个参数传递给NSKeyedArchiver实例.
- privateItems数组发送encodeWithCoder:消息给它包含的每一个对象,并传递同一个NSKeyedArchiver实例(怎么传递的?就是作为encodeWithCoder:的参数传递的),这样每一个BNRItem编码它们的实例变量到同一个NSKeyedArchiver实例中了.
- NSKeyedArchiver写数据到指定路径.
解归档方法①:
NSData *data = [[NSMutableData alloc] initWithContentsOfFile:kSavePath];
if (data)
{
NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
drawView = [unarchiver decodeObjectForKey:@"drawView"];
[unarchiver finishDecoding];
}
解归档方法②:
_privateItems = [NSKeyedUnarchiver unarchiveObjectWithFile:[self itemArchivePath]];
注:这个kSavePath,是../xxx.archive
xq注:从归档可以看出,对象是被序列化为NSData二进制数据了。以后尽量用方法②吧,简便些.
4、<NSCopying>协议
实现<NSCopying>协议
#pragma mark- copy协议
- (id)copyWithZone:(NSZone *)zone
{
BNRLine *copy = [[[self class] allocWithZone:zone] init];//这里使用[self class],是因为接收消息的可能是BNRLine的子类。
copy.startPoint = self.startPoint;
copy.endPoint = self.endPoint;
return copy;
}
归档原理
归档一个对象意味着记录它所有的属性并把它们保存到文件系统中.解归档就是从归档后的数据中重新创建对象.一个类的实例如果要能够被归档和解归档,那么这个类必须遵循NSCoding协议并实现协议里的两个方法.
- (void)encodeWithCoder:(NSCoder *)aCoder
- (id)initWithCoder:(NSCoder *)aDecoder
当一个BNRItem对象被发送encodeWithCoder:
消息时,encodeWithCoder:
方法会encode所有BNRItem对象的属性到NSCoder对象(就是方法中的那个参数aCoder)中去--所以该代理方法里面我们需要对对象的每个属性进行key-value编码.当保存的时候,你将使用NSCoder写一个数据流.这个数据流将会被保存到文件系统里.这个数据流格式是键值对.
当一个对象A被encoded的时候(即它作为encodeObject:forKey:
方法的第一个参数时),对象A会被发送- (void)encodeWithCoder:(NSCoder *)aCoder
消息.在执行对象A的- (void)encodeWithCoder:(NSCoder *)aCoder
方法时,对象A又使用encodeObject:forKey:
编码它的实例变量.因此,编码一个对象是一个递归的过程.为了能够编码,递归过程中那些对象必须都遵循了NSCoding协议.
归档之前是什么类型的对象,解归档之后就是什么类型的对象..