OC学习篇之---归档和解挡
今天我们来看一下OC中的一个重要知识点:归档
OC中的归档就是将对象写入到一个文件中,Java中的ObjectInputStream和ObjectOutputStream来进行操作的。当然在操作的这些对象都是需要实现一个接口:Serializable,同样的OC中操作的对象也是需要实现一个协议的,后面会说到。
一、已有类型的归档和解档
首先来看一个简单的例子:
- //
- // main.m
- // 33_ObjectToFile
- //
- // Created by jiangwei on 14-10-13.
- // Copyright (c) 2014年 jiangwei. All rights reserved.
- //
- #import <Foundation/Foundation.h>
- //归档:将一个对象写到文件中
- int main(int argc, const charchar * argv[]) {
- @autoreleasepool {
- //第一种形式:归档对象
- //对象----》文件
- /*
- NSArray *array = [NSArray arrayWithObjects:@"zhang",@"wangwu",@"lisi",nil];
- NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
- BOOL success = [NSKeyedArchiver archiveRootObject:array toFile:filePath];
- if(success){
- NSLog(@"保存成功");
- }
- */
- /*解归档
- NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
- id 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];
- [archiver release];
- 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 = [unarchiver decodeObjectForKey:@"scope"];
- NSLog(@"%d",value);
- }
- return 0;
- }
1、归档
- //第一种形式:归档对象
- //对象----》文件
- NSArray *array = [NSArray arrayWithObjects:@"zhang",@"wangwu",@"lisi",nil];
- NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
- BOOL success = [NSKeyedArchiver archiveRootObject:array toFile:filePath];
- if(success){
- NSLog(@"保存成功");
- }
我们这里将一个NSArray对象写入到一个文件中。
这里说到了创建一个文件的方法:
- NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
我们可以打印一下filePath的值:
NSHomeDirectory()返回的就是当前用户路径
我们查看一下array.src的内容:
我们看到内容是乱的,但是我们貌似还是能看到一点,比如wangwu/lisi等字眼,说明在归档的时候并没有深入的加密。
2、解档
- //解归档
- NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
- id array = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
- NSLog(@"%@",array);
解档也是很简单的,就是返回一个对象,不过这里用了id类型的,因为读出来也不确定是哪种类型的。
3、对多个对象进行归档到一个文件
- //第二种方式
- //第一种方式的缺陷是一个对象归档成一个文件
- //但是第二种方式,多个对象可以归档成一个文件
- NSArray *array = [NSArray arrayWithObjects:@"zhangsan",@"lisi", nil 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];
- [archiver release];
- NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
- BOOL success = [data writeToFile:filePath atomically:YES];
- if(success){
- NSLog(@"归档成功");
- }
多个对象归档的话,这里要用到一个类:NSMutableData和NSData,他们两的区别很简单,一个是可变的,一个是不可变的。然后这里还创建了一个归档器:NSKeyedArchiver,这个类负责进行指定类型的编码操作,然后将数据填充到NSMutableData类。归档的时候对每个类型对象用一个key进行对应,这个NSData和NSDirctionary很类似了。
4、对多个对象进行解档操作
- 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 = [unarchiver decodeObjectForKey:@"scope"];
- NSLog(@"%d",value);
我们可以将文件解档出一个NSData对象,然后可以通过key去获取指定的类型对象
二、自定义类型的归档和解档
上面说到了已有类型的归档和解档,下面来看一下自定义类型的归档和解档操作,在开始的时候也说了,如果自定义的类型可以进行归档和解档的话,必须实现一个协议:NSCoding
不多说了,下面来直接看代码解释:
Person.h
- //
- // Person.h
- // 34_ArchiveProtocol
- //
- // Created by jiangwei on 14-10-13.
- // Copyright (c) 2014年 jiangwei. All rights reserved.
- //
- #import <Foundation/Foundation.h>
- //类只有实现NSCoding协议才能归档
- @interface Person : NSObject<NSCoding>
- @property(nonatomic,copy)NSString *name;
- @property(nonatomic,assign)NSInteger age;
- @property(nonatomic,retain)NSArray *apples;
- - (NSString *)description;
- @end
这里自定义了一个Person类型,实现了NSCoding协议,然后他有三个属性,这里我们看到有新的方法去定义属性,这个后面说到内存管理的时候在详细说明。
Person.m
- //
- // Person.m
- // 34_ArchiveProtocol
- //
- // Created by jiangwei on 14-10-13.
- // Copyright (c) 2014年 jiangwei. All rights reserved.
- //
- #import "Person.h"
- @implementation Person
- //解归档的时候调用
- //也是一个初始化方法
- - (id)initWithCoder:(NSCoder *)aDecoder{
- NSLog(@"initWithCoder");
- self = [super init];
- if(self != nil){
- /*
- _name = [aDecoder decodeObjectForKey:@"name"];
- _age = [aDecoder decodeObjectForKey:@"age"];
- _apples = [aDecoder decodeObjectForKey:@"apples"];
- */
- //一般我们将key定义成宏,这样就不会出错
- _name = [[aDecoder decodeObjectForKey:@"name"] copy];
- self.age = [aDecoder decodeObjectForKey:@"age"];
- self.apples = [aDecoder decodeObjectForKey:@"apples"];
- }
- return self;
- }
- //归档时调用此方法
- - (void)encodeWithCoder:(NSCoder *)aCoder{
- NSLog(@"encodeWithCoder");
- [aCoder encodeObject:_name forKey:@"name"];//一般key和属性名是取一样的
- [aCoder encodeInteger:_age forKey:@"age"];
- [aCoder encodeObject:_apples forKey:@"apples"];
- }
- - (NSString *)description{
- NSString *string = [NSString stringWithFormat:@"name=%@,age=%d,apples=%@",_name,_age,_apples];
- return string;
- }
- @end
在Person.m文件中,我们需要实现协议中的两个方法:
initWithCoder
encodeWithCoder
这两个方法一个是用于归档操作时会调用的方法,还有一个是用于解档操作时会调用的方法
1、解档的时候用到的方法
- - (id)initWithCoder:(NSCoder *)aDecoder{
- NSLog(@"initWithCoder");
- self = [super init];
- if(self != nil){
- /*
- _name = [aDecoder decodeObjectForKey:@"name"];
- _age = [aDecoder decodeObjectForKey:@"age"];
- _apples = [aDecoder decodeObjectForKey:@"apples"];
- */
- //一般我们将key定义成宏,这样就不会出错
- _name = [[aDecoder decodeObjectForKey:@"name"] copy];
- self.age = [aDecoder decodeObjectForKey:@"age"];
- self.apples = [aDecoder decodeObjectForKey:@"apples"];
- }
- return self;
- }
这个是一个初始化的方法,同时他也是一个解档操作时会调用的方法,所以在这里我们既要写一下初始化方法的特定代码,还要写上解档的代码,这里主要看解档的代码
其实很简单,就是对属性重新写一下值,然后对每个属性指定一个key就可以了。这个有点类似于Android中的Parcel
(这里我们看到,在解档name属性的时候,用到了copy的一个方法,这个在后面会说到,有浅拷贝和深拷贝之分)
2、归档的时候用到的方法
- //归档时调用此方法
- - (void)encodeWithCoder:(NSCoder *)aCoder{
- NSLog(@"encodeWithCoder");
- [aCoder encodeObject:_name forKey:@"name"];//一般key和属性名是取一样的
- [aCoder encodeInteger:_age forKey:@"age"];
- [aCoder encodeObject:_apples forKey:@"apples"];
- }
归档和解档的操作正好相反的,但是要注意的是:他们属性的key一定要保持一致
3、重写description方法
- - (NSString *)description{
- NSString *string = [NSString stringWithFormat:@"name=%@,age=%d,apples=%@",_name,_age,_apples];
- return string;
- }
在之前的文章中我说道过,我们在使用NSLog方法打印对象的值的时候,其实是调用对象的description方法,而这个方法是NSObject类中的,我们可以重写他,这样我们就可以打印我们想要的信息了。和Java中的toString方法一样。
下面就来看一下使用方法了
main.m
- //
- // main.m
- // 34_ArchiveProtocol
- //
- // Created by jiangwei on 14-10-13.
- // Copyright (c) 2014年 jiangwei. All rights reserved.
- //
- #import <Foundation/Foundation.h>
- #import "Person.h"
- int main(int argc, const charchar * argv[]) {
- @autoreleasepool {
- Person *p = [[Person alloc] init];
- p.name = @"张三";
- p.age = 20;
- p.apples = @[@"iphone",@"ipad"];
- //归档
- NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"person.archiver"];
- BOOL success = [NSKeyedArchiver archiveRootObject:p toFile:filePath];
- if(success){
- NSLog(@"归档成功");
- }
- //解归档
- Person *person = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
- NSLog(@"%@",person);
- }
- return 0;
- }
我们可以看到,使用起来是很简单的和上面的方式一样,运行结果:
看到了,我们自定义的description方法,打印了我们自己想要的结果~~
总结
这一篇文章我们就说了OC中的归档和解档的相关概念和操作,其实说白了就是将对象写入到文件,和从文件中读取对象。
OC学习篇之---归档和解挡的更多相关文章
- (转载)OC学习篇之---归档和解挡
前几篇文章说到了OC中的Foundation框架,今天我们来看一下OC中的一个重要知识点:归档 OC中的归档就是将对象写入到一个文件中,Java中的ObjectInputStream和ObjectOu ...
- OC学习16——对象归档
转载自 OC学习篇之---归档和解挡 OC中的归档就是将对象写入到一个文件中,Java中的ObjectInputStream和ObjectOutputStream来进行操作的.当然在操作的这些对象都 ...
- OC学习篇之---总结和学习目录
今天终于把OC的基础知识学习完了,但是这些知识只是最基础的,还有很多高级知识,这个可能需要后面慢慢的去学习才能体会到.下面就是这次学习OC的目录教程,如果大家发现有什么不正确的地方,请指正,小弟是新生 ...
- OC学习篇之---单例模式
在之前的一片文章中介绍了对象的拷贝相关知识:http://blog.csdn.net/jiangwei0910410003/article/details/41926531,今天我们来看一下OC中的单 ...
- OC学习篇之---文件的操作
今天我们来介绍OC中文件操作,在之前的文章中,已经接触到了文件的创建了,但是那不是很具体和详细,这篇文章我们就来仔细看一下OC中是如何操作文件的: 第一.首先来看一下本身NSString类给我们提供了 ...
- (转载)OC学习篇之---概述
前言 终于开启了OC的学习篇了,之前由于工作上的事,学习就一直搁浅了,不过最近由于各种原因,感觉必须要开启iOS的开发旅程了,不然就老了.因为之前一直是做Android的,所以学习iOS来就没那么费劲 ...
- 【IOS学习基础】归档和解档
一.归档介绍 1.归档是指用某种格式来保存一个或多个对象,以便以后还原这些对象的过程.归档是将数据持久化的一种方式(所谓数据持久化,就是指在IOS开发过程中,将数据保存到本地,能够让程序的运行更加流畅 ...
- (转载)OC学习篇之---Foundation框架中的其他类(NSNumber,NSDate,NSExcetion)
前一篇说到了Foundation框架中的NSDirctionary类,这一一篇来看一下Foundation的其他常用的类:NSNumber,NSDate,NSException. 注:其实按照Java ...
- (转载)OC学习篇之---类的三大特性:封装,继承,多态
之前的一片文章介绍了OC中类的初始化方法和点语法的使用,今天来继续学习OC中的类的三大特性,我们在学习Java的时候都知道,类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今 ...
随机推荐
- 使程序在Linux下后台运行 (关掉终端继续让程序运行的方法)
你是否遇到过这样的情况:从终端软件登录远程的Linux主机,将一堆很大的文件压缩为一个.tar.gz文件,连续压缩了半个小时还没有完成,这时,突然你断网了,你登录不上远程Linux主机了,那么前面的半 ...
- mvn本地执行java程序
mvn -f pom.xml compile exec:java -Dexec.classpathScope=compile -Dexec.mainClass=storm.starter.WordCo ...
- 谈一谈Nginx的强大
什么是Nginx? Nginx是一款高性能,开源的,支持高并发而轻量级的Web服务器,同时也是具有反向代理服务器及电子邮件(IMAP/POP3)的代理服务器. 基于REST架构风格,并且以统一资源描述 ...
- 【习题 8-9 1613】 K-Graph Oddity
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 感觉最大度数|1就是最多需要的个数了. 就贪心一下. 然后模拟染色的过程就可以了. (贪心染色就可以了 (看看周围哪个颜色没有,就用 ...
- ORA-01653 无法在表空间扩展的解决办法 -- 增加表空间大小或给表空间增加数据文件
转自原文 ORA-01653 无法在表空间扩展的解决办法 -- 增加表空间大小或给表空间增加数据文件 当前系统的数据量越来越大的,昨天还运行正常的数据库,突然无法使用了.经过定位发现是"OR ...
- C++异常实现与longjmp, setjmp,栈指针EBP, Active Record
这篇讲的不错: http://blog.csdn.net/smstong/article/details/50728022 首先Active Record 然后EBP,ESP等指针 2 通过setjm ...
- ArcGIS中数据之间的转换接口IFeatureDataConverter2
之前我写过一篇文章关于ArcGIS各种空间数据格式之间转换的通用方法:ArcGIS中sde,mdb,shp数据之间的转换.这里使用的主要接口方法就是用到了IFeatureDataConverter接口 ...
- 【hadoop之翊】——CentOS6.5 Linux上面编译Hadoop2.4源代码
今天来说说编译hadoop源代码的事情吧~ 1.首先下载源代码 地址:http://mirror.bit.edu.cn/apache/hadoop/common/hadoop-2.4.0/ 2.加压t ...
- Elasticsearch中JAVA API的使用
1.Elasticsearch中Java API的简介 Elasticsearch 的Java API 提供了非常便捷的方法来索引和查询数据等. 通过添加jar包,不需要编写HTTP层的代码就可以开始 ...
- 没有killall命令的解决方法
没有killall命令的解决方法 -bash: killall: command not found https://www.byte128.com/archives/231.html 执行killa ...