归档

1、什么叫归档
归档:即序列化。任何对象都可以遵循协议进行归档。通过对数据模型对象进行归档可以轻松将复杂的对象写入文件,然后再从中读取它们。
另外,尽管对归档的使用没有严格要求,但还有一个协议应该与一起实现,那就是协议。后者允许复制对象,这会让你在使用数据模型对象时具备了较大的灵活性。

@protocol NSCoding

- (void)encodeWithCoder:(NSCoder *)aCoder;
- (id)initWithCoder:(NSCoder *)aDecoder; // NS_DESIGNATED_INITIALIZER

@end

2、如何归档
假设B为A的子类,A、B都遵循协议,那么B类在实现协议时,需要调用父类的对应方法,如果A没有遵循协议那么B类在实现协议时就不需要调用父类的对应方法了。如下:

- (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];

这种方法和上面那种方法原理上是一样的,方法②应该是苹果提供的一种简便方法.
工作原理如下:

  1. 这个方法首先创建了一个NSKeyedArchiver实例.(NSKeyedArchiver类是抽象类NSCoder的一个子类,是一个具体类)
  2. self.privateItems被发送消息encodeWithCoder:,并且被作为一个参数传递给NSKeyedArchiver实例.
  3. privateItems数组发送encodeWithCoder:消息给它包含的每一个对象,并传递同一个NSKeyedArchiver实例(怎么传递的?就是作为encodeWithCoder:的参数传递的),这样每一个BNRItem编码它们的实例变量到同一个NSKeyedArchiver实例中了.
  4. 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、协议
实现协议

#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协议.
归档之前是什么类型的对象,解归档之后就是什么类型的对象..

你可能感兴趣的:(归档)