一、已有类型的归档和解档

首先来看一个简单的例子:

 //第一方式:归档对象
//对象-->文件
NSArray *array = [NSArray arrayWithObjects:@"zhang",@"wangwu",@"lisi",nil]; // NSHomeDirectory 获取根目录 stringByAppendingPathComponent 添加储存的文件名 NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
BOOL success = [NSKeyedArchiver archiveRootObject:array toFile:filePath];
if(success){
NSLog(@"保存成功");
}else {
NSLog(@"未保存");
}
// 解归档
array = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
NSLog(@"%@",array);

//第二种方式

//第一种方式的缺陷是一个对象归档成一个文件

//但是第二种方式,多个对象可以归档成一个文件

NSArray *array = [NSArray arrayWithObjects:@"zhangsan",@"lisi", nil];

NSMutableData *data = [NSMutableData data];

NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];

//编码

[archiver encodeObject:array forKey:@"array"];

[archiver encodeInt:100 forKey:@"scope"];

[archiver encodeObject:@"jack" forKey:@"name"];

//完成编码,将上面的归档数据填充到data中,此时data中已经存储了归档对象的数据

[archiver finishEncoding];

NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];

BOOL success = [data writeToFile:filePath atomically:YES];

if(success){

NSLog(@"归档成功");

}

//  对多个对象进行解档操作

/*

NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];

//读取归档数据

NSData *data = [[NSData alloc] initWithContentsOfFile:filePath];

//创建解归档对象,对data中的数据进行解归档

NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];

//解归档

NSArray *array = [unarchiver decodeObjectForKey:@"array"];

NSLog(@"%@",array);

int value = (int)[unarchiver decodeObjectForKey:@"scope"];

NSLog(@"%d",value);

*/

  • 归档  下面这段代码是将一个NSArray对象写入到一个文件中。
//对象-->文件
NSArray *array = [NSArray arrayWithObjects:@"zhang",@"wangwu",@"lisi",nil]; // NSHomeDirectory 获取根目录 stringByAppendingPathComponent 添加储存的文件名 NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
BOOL success = [NSKeyedArchiver archiveRootObject:array toFile:filePath];
if(success){
NSLog(@"保存成功");
}else {
NSLog(@"未保存");

下面代码是创建一个文件的方法

NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  • 解档  下面代码是解档就是返回一个对象
array = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
NSLog(@"%@",array);
  • 对多个对象进行归档到一个文件
//第二种方式
//第一种方式的缺陷是一个对象归档成一个文件
//但是第二种方式,多个对象可以归档成一个文件
NSArray *array = [NSArray arrayWithObjects:@"zhangsan",@"lisi", nil];
NSMutableData *data = [NSMutableData data];
NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
//编码
[archiver encodeObject:array forKey:@"array"];
[archiver encodeInt: forKey:@"scope"];
[archiver encodeObject:@"jack" forKey:@"name"]; //完成编码,将上面的归档数据填充到data中,此时data中已经存储了归档对象的数据
[archiver finishEncoding]; NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
BOOL success = [data writeToFile:filePath atomically:YES];
if(success){
NSLog(@"归档成功");
}

多个对象归档的话,这里要用到一个类:NSMutableData和NSData,他们两的区别很简单,一个是可变的,一个是不可变的。然后这里还创建了一个归档器:NSKeyedArchiver,这个类负责进行指定类型的编码操作,然后将数据填充到NSMutableData类。归档的时候对每个类型对象用一个key进行对应,这个NSData和NSDirctionary很类似了。

  • 对多个对象进行解档操作
NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
//读取归档数据
NSData *data = [[NSData alloc] initWithContentsOfFile:filePath]; //创建解归档对象,对data中的数据进行解归档
NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data]; //解归档
NSArray *array = [unarchiver decodeObjectForKey:@"array"];
NSLog(@"%@",array); int value = (int)[unarchiver decodeObjectForKey:@"scope"];
NSLog(@"%d",value);

我们可以将文件解档出一个NSData对象,然后可以通过key去获取指定的类型对象

二、自定义类型的归档和解档

上面说到了已有类型的归档和解档,下面来看一下自定义类型的归档和解档操作,在开始的时候也说了,如果自定义的类型可以进行归档和解档的话,必须实现一个协议:NSCoding

Student.h

#import <Foundation/Foundation.h>

// 类只有实现NSCoding协议才能归档
@interface Student : NSObject<NSCoding> @property(copy,nonatomic)NSString *name;
@property(assign,nonatomic)int age;
@property(strong,nonatomic) NSString *adder; @end

这里自定义了一个Student类型,实现了NSCoding协议,然后他有三个属性,这里我们看到有新的方法去定义属性

Student.m

#import "Student.h"

@implementation Student

// 归档时调用   也是一个初始化

- (instancetype)initWithCoder:(NSCoder *)aDecoder
{
NSLog(@"initWithCoder");
self = [super init];
if (self!=nil) {
//一般我们将key定义成宏,这样就不会出错
_name = [[aDecoder decodeObjectForKey:@"name"] copy];
self.age = (int)[aDecoder decodeIntegerForKey:@"age"];
_adder=[aDecoder decodeObjectForKey:@"adder"];
}
return self;
} // 归档时调用此方法
- (void)encodeWithCoder:(NSCoder *)aCoder{
NSLog(@"encodeWithCoder");
[aCoder encodeObject:_name forKey:@"name"];//一般key和属性名是取一样的
[aCoder encodeInteger:_age forKey:@"age"];
[aCoder encodeObject:_adder forKey:@"adder"]; } // 描述方法
- (NSString *)description
{
return [NSString stringWithFormat:@"name=%@,age=%d,adder=%@", _name,_age,_adder];
} @end

在Person.m文件中,我们需要实现协议中的两个方法:

initWithCoder

encodeWithCoder

这两个方法一个是用于归档操作时会调用的方法,还有一个是用于解档操作时会调用的方法

1、解档的时候用到的方法

- (instancetype)initWithCoder:(NSCoder *)aDecoder
{
NSLog(@"initWithCoder");
self = [super init];
if (self!=nil) {
//一般我们将key定义成宏,这样就不会出错
_name = [[aDecoder decodeObjectForKey:@"name"] copy];
self.age = (int)[aDecoder decodeIntegerForKey:@"age"];
_adder=[aDecoder decodeObjectForKey:@"adder"];
}
return self;
}

这个是一个初始化的方法,同时他也是一个解档操作时会调用的方法,所以在这里我们既要写一下初始化方法的特定代码,还要写上解档的代码,这里主要看解档的代码

其实很简单,就是对属性重新写一下值,然后对每个属性指定一个key就可以了。

2、归档的时候用到的方法

// 归档时调用此方法
- (void)encodeWithCoder:(NSCoder *)aCoder{
NSLog(@"encodeWithCoder");
[aCoder encodeObject:_name forKey:@"name"];//一般key和属性名是取一样的
[aCoder encodeInteger:_age forKey:@"age"];
[aCoder encodeObject:_adder forKey:@"adder"]; }

归档和解档的操作正好相反的,但是要注意的是:他们属性的key一定要保持一致

3、重写description方法

在之前的文章中我说道过,我们在使用NSLog方法打印对象的值的时候,其实是调用对象的description方法,而这个方法是NSObject类中的,我们可以重写他,这样我们就可以打印我们想要的信息了。和Java中的toString方法一样。

下面就来看一下使用方法了

#import "ViewController.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
[super viewDidLoad];
Student *stu=[Student new];
stu.name=@"张三";
stu.age=;
stu.adder=@"北京"; // 归档
NSString *filePath=[NSHomeDirectory() stringByAppendingPathComponent:@"message.plist"];
NSLog(@"%@",filePath);
BOOL bol=[NSKeyedArchiver archiveRootObject:stu toFile:filePath]; if (bol) {
NSLog(@"归档成功");
}else{
NSLog(@"归档成功");
} // 解归档
Student *stu1=[NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
NSLog(@"%@",stu1);
} - (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
} @end

我们可以看到,使用起来是很简单的和上面的方式一样,运行结果:

看到了,我们自定义的description方法,打印了我们自己想要的结果~~

iOS--归档和解档(Archiver)、(UnArchiver)的更多相关文章

  1. 【IOS学习基础】归档和解档

    一.归档介绍 1.归档是指用某种格式来保存一个或多个对象,以便以后还原这些对象的过程.归档是将数据持久化的一种方式(所谓数据持久化,就是指在IOS开发过程中,将数据保存到本地,能够让程序的运行更加流畅 ...

  2. iOS基础知识之归档和解档

    归档和解档:即将数据写入文件和从文件中读取数据. 此处以plist文件为例说明, 一.plist文件使用时的注意事项: 1.plist文件中仅支持写入Array,Dictionary,Boolean, ...

  3. runtime之归档和解档

    IOS开发之NSCoding协议(使用runtime)近期学习IOS的runtime库,然后看到之前写的NSCoding协议有点复杂,如果属性少还好,如果100多个属性,则会显得麻烦.下面使用常规方式 ...

  4. OC 归档和解档

    #import <Foundation/Foundation.h> #define PATH @"/Users/mac/Desktop/file.txt" int ma ...

  5. 利用Runtime对Ivar实例变量进行共用的归档和解档方式

    一.介绍 在OC中每一个对象持有的变量都是实例变量,实例变量包括成员变量和属性变量,在runtime中用Ivar表示对象的实例变量.其实,runtime源码中可以看到,Ivar也是一个结构体(基本上在 ...

  6. 归档和解档配合NSFile存储数据

    NSString *Name = @"yc"; //第一个常量NSDocumentDirectory表示正在查找沙盒Document目录的路径(如果参数为NSCachesDirec ...

  7. iOS 浅复制、深复制、完全复制的知识点梳理验证(附加归档解档)

    在之前转载的一片文章中,文中对浅复制和深复制进行了详细的解读,同时还提到了深复制(one-level-deep copy).完全复制(true copy)的概念,并指出iOS开发中的深复制是单层深赋值 ...

  8. iOS 浅赋值、深复制、全然复制的知识点梳理验证(附加归档解档)

    写于前: 在之前转载的一片文章中.文中对浅复制和深复制进行了具体的解读,同一时候还提到了深复制(one-level-deep copy).全然复制(true copy)的概念,并指出iOS开发中的深复 ...

  9. iOS 数据存储 - 归档和解归档

    这里的归档主要是用于自定义类的归档和解档.我们这里使用NSKeyedArchiver和NSKeyedUnarchiver来归档和解档. 注意:自己定义的类需要实现<NSCoding>,如: ...

随机推荐

  1. Js位置与大小(1)——正确理解和运用与尺寸大小相关的DOM属性

    在web开发中,不可避免遇到要计算元素大小以及位置的问题,解决这类问题的方法是利用DOM提供的一些API结合兼容性处理来,所有内容大概分3篇左右的文章的来说明.本文作为第一篇,介绍DOM提供的与尺寸大 ...

  2. 原生JS 获取浏览器、窗口、元素等尺寸的方法及注意事项

    一.通过浏览器获得屏幕的尺寸 screen.width screen.height screen.availHeight //获取去除状态栏后的屏幕高度 screen.availWidth //获取去 ...

  3. 【原创】kafka controller源代码分析(一)

    Kafka集群中的一个broker会被作为controller负责管理分区和副本的状态以及执行类似于重分配分区之类的管理任务.如果当前的controller失败了,会从剩下的broker中选出新的co ...

  4. js jquery 异常

    1.0 TypeError: $(...).carousel is not a function原因:可能是没有引用或者引用了多个jquery:<script src="~/js/jq ...

  5. 实现了IEnumerable接口的GetEnumerator 即可使用 Foreach遍历,返回一个IEnumerator对象

    #region 程序集 mscorlib.dll, v4.0.0.0 // C:\Program Files (x86)\Reference Assemblies\Microsoft\Framewor ...

  6. 用c#开发微信 (4) 基于Senparc.Weixin框架的接收事件推送处理 (源码下载)

    本文讲述使用Senparc.Weixin框架来快速处理各种接收事件推送.这里的消息指的是传统的微信公众平台消息交互,微信用户向公众号发送消息后,公众号回复消息给微信用户.包括以下类型: 1 subsc ...

  7. Asp.net Mvc模块化开发之“部分版本部分模块更新(上线)”

    项目开发从来就不是一个简单的问题.更难的问题是维护其他人开发的项目,并且要修改bug.如果原系统有重大问题还需要重构. 怎么重构系统不是本文探讨的问题,但是重构后如何上线部署和本文关系密切.这个大家可 ...

  8. 通用easyui查询页面组件

    easyui查询页面组件使用指南 本组件开发需求:信息系统的查询页面基本是包括:搜索区域,列表显示区域,按钮条. 1.录入一个查询语句(如:select * from Strudents),录入列表显 ...

  9. 部署Eclipse中的Web项目到Tomcat服务器运行

    用Eclipse开发Web项目时,可以通过Tomcat服务器运行Web项目,此时Web项目被部署在[WorkSpace]\.metadata\.plugins\org.eclipse.wst.serv ...

  10. Gitblit中采用Ticket模式进行协作开发

    Git目前的代码分支管理模型中,比较主要的有Git-Flow.Github Pull Request.大家日常或多或少都在用着. 在不想安装Gitlab这种重量级的环境的情况下,如果是利用git一步步 ...