对象编码
对象编码把对象的类特征以及对象状态转化某种格式,这种格式可存储,也可在进程间传递。类的类型以及实例数据会被写入到某种字节流里面,当程序结束的时候,该字节流可进行持久化。程序启动的时候,新分配对象可以解码之前保存的自身描述,然后把自己恢复到之前运行时的状态。编码通常需和归档协作。归档将对象图转化成一种可以写进文件系统的格式(档案),解档则是在档案上执行操作。在此过程中,保存在档案的对象对需要自身进行解码。
对象编码也被Mac OS X 系统的分布式对象API用于进程间传输对象。但是归档用得最为普遍,归档和属性表相似,也是一种对象持久化机制。
如何编码解码对象
如果您期望子类实例能够进行编解码,则子类应遵循NSCoding
协议并实现 initWithCoder:
以及encodeWithCoder:
方法,这些方法会在程序归解档某个对象图的时候被调用。在encodeWithCoder:
方法中,我们会编码对象的重要实例变量,而在initWithCoder:
方法中,我们则会解码这些值,并将它们重新赋给实例变量。initWithCoder:
消息不会触发对象的初始化方法。
initWithCoder:
方法和encodeWithCoder:
方法都只需要一个NSCoder
对象作为参数,该对象用于执行实际的编解码动作。NSCoder
是个抽象类,大部分情况下,编解码对象是下述具体子类的实例:NSKeyedArchiver
,NSKeyedUnarchiver
,NSArchiver
,NSUnarchiver
。归档类声明编码对象实例变量的方法,解档类则声明解码实例变量的方法。
NSCoder
方法可作用于对象、标量、C数组、结构体、字符串以及指向这些类型的指针。编码自有类实例变量之前,请您务必要先调用超类的initWithCoder:
或者encodeWithCoder:
实现。当您从字节流解码对象时,也务必要先保持或复制这些对象,然后再把解码出来的值赋给其实例变量。
键归档和顺序归档
NSCoder
有两个具体子类,二者各自使用不同的基本实现方式。“键”归档类(NSKeyedArchiver
和NSKeyedUnarchiver
)将一个编码值和一个字符串键关联在一起。解码的时候 ,键值之间的关联保持 一致。因此, 实例变量可按任意的次序编解码。如使用另一种类型的编解码器(NSKeyedUnarchiver
和NSUnarchiver
),您需要以某种次序编码实例变量。解码的时候,您仍要使用相同的次序。顺序编解码器只应在遗留代码里面才能使用,新的子类都应使用键归档编解码器。
例子:
1.协议
- @interface untitled : NSObject <NSCoding>{
- NSString *string;
-
- }
2.实现协议
-(void) encodeWithCoder: (NSCoder *) aCoder{
//label.text = @"enconde";
NSLog(@"encode");
[aCoder encodeObject:string forKey:@"code"];
}
-(id) initWithCoder: (NSCoder *) aDecoder{
NSLog(@"initWithCoder");
if (self = [super init]) {
self.string = [aDecoder decodeObjectForKey:@"code"];
//注意self不能少,与内存管理有关
}
return (self);
}
3.归档
- NSString *path = [[NSBundle mainBundle] bundlePath];
- path = [[NSString alloc] initWithFormat:@"%@%@",path,@"/code.txt"];
- untitled *t;
- [NSKeyedArchiver archiveRootObject:coder toFile:path ];
4.解档
- NSString *path = [[NSBundle mainBundle] bundlePath];
- path = [[NSString alloc] initWithFormat:@"%@%@",path,@"/code.txt"];
- untitled *t;
- t = [NSKeyedUnarchiver unarchiveObjectWithFile:path];
前几篇文章说到了OC中的Foundation框架:http://blog.csdn.net/jiangwei0910410003/article/details/41852835,今天我们来看一下OC中的一个重要知识点:归档
OC中的归档就是将对象写入到一个文件中,Java中的ObjectInputStream和ObjectOutputStream来进行操作的。当然在操作的这些对象都是需要实现一个接口:Serializable,同样的OC中操作的对象也是需要实现一个协议的,后面会说到。
一、已有类型的归档和解档
首先来看一个简单的例子:
-
-
-
-
-
-
-
-
- #import <Foundation/Foundation.h>
-
-
- int main(int argc, const charchar * argv[]) {
- @autoreleasepool {
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
-
- NSData *data = [[NSData alloc] initWithContentsOfFile:filePath];
-
-
- 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"];
-
-
- [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];
-
-
- 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
-
-
-
-
-
-
-
-
- #import <Foundation/Foundation.h>
-
-
- @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
-
-
-
-
-
-
-
-
- #import "Person.h"
-
- @implementation Person
-
-
-
- - (id)initWithCoder:(NSCoder *)aDecoder{
- NSLog(@"initWithCoder");
- self = [super init];
- if(self != nil){
-
-
-
-
-
-
- _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"];
- [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"] 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"];
- [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
-
-
-
-
-
-
-
-
- #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中的归档和解档的相关概念和操作,其实说白了就是将对象写入到文件,和从文件中读取对象。