博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
归档 浅析
阅读量:5970 次
发布时间:2019-06-19

本文共 6454 字,大约阅读时间需要 21 分钟。

转载自:https://nezha.gitbooks.io/ios-developmentarticles/content/iOS%20archive%EF%BC%88%E5%BD%92%E6%A1%A3%EF%BC%89%E7%9A%84%E6%80%BB%E7%BB%93.html

OS 归档的记录

归档是一种很常用的文件储存方法,几乎​​任何类型的对象都能够被归档储存(实际上是一种文件保存的形式),浏览网上的一些资料后,并结合自己的一些经验,总结成此文。

一、使用archiveRootObject进行简单的归档、

使用NSKeyedArichiver进行归档、NSKeyedUnarchiver进行接档,这种方式会在写入、读出数据之前对数据进行序列化、反序列化操作归档:

NSString *homeDictionary = NSHomeDirectory();//获取根目录  NSString *homePath = [homeDictionary stringByAppendingPathComponent:@"atany.archiver"];//添加储存的文件名  BOOL flag = [NSKeyedArchiver archiveRootObject:@”归档” toFile:homePath];//归档一个字符串

这种方式可以对字符串、数字等进行归档,当然也可以对NSArray与NSDictionary进行归档。返回值Flag标志着是否归档成功,YES为成功,NO为失败。

接档:

[NSKeyedUnarchiver unarchiveObjectWithFile:homePath]

二、对多个对象的归档

同样是使用NSKeyedArchiver进行归档,不同的是同时归档多个对象,这里我们举例放入了一个CGPoint点、字符串、整数(当然很多类型都可以的,例如UIImage、float等等),使用encodeXXX方法进行归档,最后通过writeToFile方法写入文件。

归档:写入数据

//准备数据  CGPoint point = CGPointMake(1.0, 2.0);  NSString *info = @"坐标原点";  NSInteger value = 10;  NSString *multiHomePath = [NSHomeDirectory() stringByAppendingPathComponent:@"multi.archiver"];  NSMutableData *data = [[NSMutableData alloc]init];  NSKeyedArchiver *archvier = [[NSKeyedArchiver alloc]initForWritingWithMutableData:data];  //对多个对象进行归档  [archvier encodeCGPoint:point forKey:@"kPoint"];  [archvier encodeObject:info forKey:@"kInfo"];  [archvier encodeInteger:value forKey:@"kValue"];  [archvier finishEncoding];  [data writeToFile:multiHomePath atomically:YES];

接档:从路径中获得数据构造NSKeyedUnarchiver实例,使用decodeXXXForKey方法获得文件中的对象。

SMutableData *dataR = [[NSMutableData alloc]initWithContentsOfFile:multiHomePath];  NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc]initForReadingWithData:dateR];  CGPoint pointR = [unarchiver decodeCGPointForKey:@"kPoint"];  NSString *infoR = [unarchiver decodeObjectForKey:@"kInfo"];  NSInteger valueR = [unarchiver decodeIntegerForKey:@"kValue"];  [unarchiver finishDecoding];  NSLog(@"%f,%f,%@,%d",pointR.x,pointR.y,infoR,valueR);

可以看出对多个对象进行归档还是挺方便的,这里又出现一个问题,这里的对象都是基本类型数据,那么怎么对自己定义类生成的实例对象进行归档呢?

三、对自定义对象进行归档

自定义对象,应用范围很广,因为它对应着MVC中的Model层,即实体类。在程序中,我们会在Model层定义很多的entity,例如User,Teacher。。

那么对自定义对象的归档显得重要的多,因为很多情况下我们需要在Home键之后保存数据,在程序恢复时重新加载,那么,归档便是一个好的选择。

首先我们需要,自定义一个实体类,Archive。

Archive.h

#import 
@interface Archive : NSObject @property (copy,nonatomic) NSString *name; @property NSInteger age; @property (copy,nonatomic) NSString *address; @property (copy,nonatomic) UIImage *photo; @end

Archive.m

#import "Archive.h"  #define kNameKey @"NameKey"  #define kAgeKey @"AgeKey"  #define kAddress @"AddressKey"  #define kPhotoKey @"PhotoKey"  @implementation Archive  @synthesize name = _name;  @synthesize age = _age;  @synthesize address = _address;  @synthesize photo = _photo;  #pragma mark - NSCoding  - (void)encodeWithCoder:(NSCoder *)aCoder {      [aCoder encodeObject:_name forKey:kNameKey];      [aCoder encodeInteger:_age forKey:kAgeKey];      [aCoder encodeObject:_address forKey:kAddress];      [aCoder encodeObject:_photo forKey:kPhotoKey];  }  - (id)initWithCoder:(NSCoder *)aDecoder {      if (self = [super init]) {          _name = [aDecoder decodeObjectForKey:kNameKey];          _age = [aDecoder decodeIntegerForKey:kAgeKey];          _address = [aDecoder decodeObjectForKey:kAddress];          _photo = [aDecoder decodeObjectForKey:kPhotoKey];      }      return self;  }  #pragma mark - NSCoping  - (id)copyWithZone:(NSZone *)zone {      Archive *copy = [[[self class] allocWithZone:zone] init];      copy.name = [self.name copyWithZone:zone];      copy.age = self.age;      copy.address = [self.address copyWithZone:zone];      copy.photo = self.photo;      return copy;  }  @end

Archive类有四个字段(名字、年纪、地址、头像),除了年纪为整型之外,其他的都看作Object。

【注】:要将一个自定义的类进行归档,那么类里面的每个属性都必须是可以被归档的,如果是不能归档的类型,我们可以把他转化为NSValue进行归档,然后在读出来的时候在转化为相应的类。

Archive实现了三个委托方法1)encodeWithCoder: 2)initWithCoder: 3)copyWithZone:

1)encodeWithCoder

Encodes the receiverusing a given archiver

通过一个给定的archiver把消息接收者进行编码。

当接收到encodeObject消息的时候,类终端encodeWithCoder方法被调用。

2)initWithCoder

Returns an objectinitialized from data in a given unarchiver. (required)

从一个给定unarchiver的数据中返回一个初始化对象。

3)copyWithZone

Returnsa new instance that's a copy of the receiver

返回消息接收者的一个复制的新实例。

SDK的概念就是这样,下面看看这个自定义类归档的具体代码,其实和多个对象的归档是一样的。。。

归档:

//保存图片与归档  - (IBAction)save:(id)sender {      //准备数据      NSString *name = @"小杨在玩iOS";      NSInteger age = 22;      NSString *address = @"你猜我在哪~";      UIImage *photo = [UIImage imageNamed:@"loginman.jpg"];      //存储数据到类      Archive *archivingData = [[Archive alloc] init];      archivingData.name = name;      archivingData.age = age;      archivingData.address = address;      archivingData.photo = photo;      //归档      NSMutableData *data = [[NSMutableData alloc] init];      NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];      [archiver encodeObject:archivingData forKey:kArchivingDataKey]; // archivingDate的encodeWithCoder  方法被调用      [archiver finishEncoding];      //写入文件      [data writeToFile:self.archivingFilePath atomically:YES];  }

接档:

- (IBAction)loadArchive:(id)sender {      NSData *data = [[NSMutableData alloc] initWithContentsOfFile:self.archivingFilePath];      NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];      //获得类      Archive *archivingData = [unarchiver decodeObjectForKey:kArchivingDataKey];// initWithCoder方法被调用      [unarchiver finishDecoding];      //读取的数据      NSString *name = archivingData.name;      NSInteger age = archivingData.age;      NSString *address = archivingData.address;      self.imageView.image = archivingData.photo;      NSLog(@"%@||%d||%@",name,age,address);  }

我们save之后loadArchive一次

打出结果为:

 

转载自:http://my.oschina.net/u/2560887/blog/603367

在面向对象的编程中,会涉及到对象的保存,对象的保存在JAVA当中称之为serialize,也叫序列化。而在cocoa当中,这个过程称之为归档。cocoa当中最最典型的序列化例子就是NIB文件的保存和读取过程。

在cocoa当中如何实现了这个功能的哪?

其实在cocoa当中是通过了NSCoding协议和NSCoder抽象类共同实现的,对于NSCoder抽象类而言,其实他只是一个抽象类,是不能实例化使用的,而真正使用的是NSCoder的子类,即NSArchiver, NSUnarchiver, NSKeyedArchiver, NSKeyedUnarchiver类的实例对象。

那么实现的关键点有两点:

1、要使的需要存储的对象实现NSCoding协议,从而使他自己满足写二进制数据的能力。即是自己具有序列化的能力。

2、使用NSCoder的子类方法,实现二进制数据的动作,如读或者写。即触发存档过程。

需要注意的一点就是说,其实对于序列化,是一个“一传十,十传百”的过程,父类的序列化,会逐层递归的将所有的子对象也进行了序列化。此外,序列化,能够序列化的是对象实例和类名,而对于方法其实是无法序列化的,因此对于序列化和非序列化的类,都需要使用共同的类定义,否则无法还原。

 

NSCoding协议的方法:

-(id)initWithCoder:(NSCoder *)coder; // 读取coder中的数据

-(void) encodeWithCoder:(NSCoder*)coder;// 向coder中写入数据

 

NSKeyArchiver的方法

+(NSData*)archivedDataWithRootObject:(id)rootObject

获取成功。

你可能感兴趣的文章
Pycharm选择pyenv安装的Python版本
查看>>
?Sized 和 Sized
查看>>
Java中如何防止内存泄漏的发生
查看>>
Java中Int转byte分析
查看>>
滑动窗口最大值的golang实现
查看>>
会计的思考(17):还原会计报表的企业个性之一
查看>>
java对象初始化顺序的简单验证
查看>>
[CF452E]Three strings
查看>>
获取指定进程所对应的可执行(EXE)文件全路径(代码)
查看>>
ORA-01722:无效数字
查看>>
搭建golang+vscode开发环境
查看>>
C#占位符
查看>>
java面试-JVM调优和参数配置
查看>>
常用的激活函数
查看>>
sqlcmd
查看>>
Excel 已经检测到"XXX.xsl"是SYLK文件,但是不能将其加载的问题
查看>>
(基础篇)PHP获取时间、时间戳的各种格式写法汇总
查看>>
浅析面向对象开发
查看>>
Hystrix降级逻辑中如何获取触发的异常
查看>>
【跃迁之路】【535天】程序员高效学习方法论探索系列(实验阶段292-2018.07.25)...
查看>>