今天我们来看一下OC中的一个重要知识点:归档

OC中的归档就是将对象写入到一个文件中,Java中的ObjectInputStream和ObjectOutputStream来进行操作的。当然在操作的这些对象都是需要实现一个接口:Serializable,同样的OC中操作的对象也是需要实现一个协议的,后面会说到。

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

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

  1. //
  2. //  main.m
  3. //  33_ObjectToFile
  4. //
  5. //  Created by jiangwei on 14-10-13.
  6. //  Copyright (c) 2014年 jiangwei. All rights reserved.
  7. //
  8. #import <Foundation/Foundation.h>
  9. //归档:将一个对象写到文件中
  10. int main(int argc, const charchar * argv[]) {
  11. @autoreleasepool {
  12. //第一种形式:归档对象
  13. //对象----》文件
  14. /*
  15. NSArray *array = [NSArray arrayWithObjects:@"zhang",@"wangwu",@"lisi",nil];
  16. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  17. BOOL success = [NSKeyedArchiver archiveRootObject:array toFile:filePath];
  18. if(success){
  19. NSLog(@"保存成功");
  20. }
  21. */
  22. /*解归档
  23. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  24. id array = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
  25. NSLog(@"%@",array);
  26. */
  27. //第二种方式
  28. //第一种方式的缺陷是一个对象归档成一个文件
  29. //但是第二种方式,多个对象可以归档成一个文件
  30. /*
  31. NSArray *array = [NSArray arrayWithObjects:@"zhangsan",@"lisi", nil];
  32. NSMutableData *data = [NSMutableData data];
  33. NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
  34. //编码
  35. [archiver encodeObject:array forKey:@"array"];
  36. [archiver encodeInt:100 forKey:@"scope"];
  37. [archiver encodeObject:@"jack" forKey:@"name"];
  38. //完成编码,将上面的归档数据填充到data中,此时data中已经存储了归档对象的数据
  39. [archiver finishEncoding];
  40. [archiver release];
  41. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  42. BOOL success = [data writeToFile:filePath atomically:YES];
  43. if(success){
  44. NSLog(@"归档成功");
  45. }
  46. */
  47. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  48. //读取归档数据
  49. NSData *data = [[NSData alloc] initWithContentsOfFile:filePath];
  50. //创建解归档对象,对data中的数据进行解归档
  51. NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
  52. //解归档
  53. NSArray *array = [unarchiver decodeObjectForKey:@"array"];
  54. NSLog(@"%@",array);
  55. int value = [unarchiver decodeObjectForKey:@"scope"];
  56. NSLog(@"%d",value);
  57. }
  58. return 0;
  59. }

1、归档

  1. //第一种形式:归档对象
  2. //对象----》文件
  3. NSArray *array = [NSArray arrayWithObjects:@"zhang",@"wangwu",@"lisi",nil];
  4. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  5. BOOL success = [NSKeyedArchiver archiveRootObject:array toFile:filePath];
  6. if(success){
  7. NSLog(@"保存成功");
  8. }

我们这里将一个NSArray对象写入到一个文件中。

这里说到了创建一个文件的方法:

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

我们可以打印一下filePath的值:

NSHomeDirectory()返回的就是当前用户路径

我们查看一下array.src的内容:

我们看到内容是乱的,但是我们貌似还是能看到一点,比如wangwu/lisi等字眼,说明在归档的时候并没有深入的加密。

2、解档

  1. //解归档
  2. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  3. id array = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
  4. NSLog(@"%@",array);

解档也是很简单的,就是返回一个对象,不过这里用了id类型的,因为读出来也不确定是哪种类型的。

3、对多个对象进行归档到一个文件

  1. //第二种方式
  2. //第一种方式的缺陷是一个对象归档成一个文件
  3. //但是第二种方式,多个对象可以归档成一个文件
  4. NSArray *array = [NSArray arrayWithObjects:@"zhangsan",@"lisi", nil nil];
  5. NSMutableData *data = [NSMutableData data];
  6. NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
  7. //编码
  8. [archiver encodeObject:array forKey:@"array"];
  9. [archiver encodeInt:100 forKey:@"scope"];
  10. [archiver encodeObject:@"jack" forKey:@"name"];
  11. //完成编码,将上面的归档数据填充到data中,此时data中已经存储了归档对象的数据
  12. [archiver finishEncoding];
  13. [archiver release];
  14. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  15. BOOL success = [data writeToFile:filePath atomically:YES];
  16. if(success){
  17. NSLog(@"归档成功");
  18. }

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

4、对多个对象进行解档操作

  1. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"array.src"];
  2. //读取归档数据
  3. NSData *data = [[NSData alloc] initWithContentsOfFile:filePath];
  4. //创建解归档对象,对data中的数据进行解归档
  5. NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
  6. //解归档
  7. NSArray *array = [unarchiver decodeObjectForKey:@"array"];
  8. NSLog(@"%@",array);
  9. int value = [unarchiver decodeObjectForKey:@"scope"];
  10. NSLog(@"%d",value);

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

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

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

不多说了,下面来直接看代码解释:

Person.h

  1. //
  2. //  Person.h
  3. //  34_ArchiveProtocol
  4. //
  5. //  Created by jiangwei on 14-10-13.
  6. //  Copyright (c) 2014年 jiangwei. All rights reserved.
  7. //
  8. #import <Foundation/Foundation.h>
  9. //类只有实现NSCoding协议才能归档
  10. @interface Person : NSObject<NSCoding>
  11. @property(nonatomic,copy)NSString *name;
  12. @property(nonatomic,assign)NSInteger age;
  13. @property(nonatomic,retain)NSArray *apples;
  14. - (NSString *)description;
  15. @end

这里自定义了一个Person类型,实现了NSCoding协议,然后他有三个属性,这里我们看到有新的方法去定义属性,这个后面说到内存管理的时候在详细说明。

Person.m

  1. //
  2. //  Person.m
  3. //  34_ArchiveProtocol
  4. //
  5. //  Created by jiangwei on 14-10-13.
  6. //  Copyright (c) 2014年 jiangwei. All rights reserved.
  7. //
  8. #import "Person.h"
  9. @implementation Person
  10. //解归档的时候调用
  11. //也是一个初始化方法
  12. - (id)initWithCoder:(NSCoder *)aDecoder{
  13. NSLog(@"initWithCoder");
  14. self = [super init];
  15. if(self != nil){
  16. /*
  17. _name = [aDecoder decodeObjectForKey:@"name"];
  18. _age = [aDecoder decodeObjectForKey:@"age"];
  19. _apples = [aDecoder decodeObjectForKey:@"apples"];
  20. */
  21. //一般我们将key定义成宏,这样就不会出错
  22. _name = [[aDecoder decodeObjectForKey:@"name"] copy];
  23. self.age = [aDecoder decodeObjectForKey:@"age"];
  24. self.apples = [aDecoder decodeObjectForKey:@"apples"];
  25. }
  26. return self;
  27. }
  28. //归档时调用此方法
  29. - (void)encodeWithCoder:(NSCoder *)aCoder{
  30. NSLog(@"encodeWithCoder");
  31. [aCoder encodeObject:_name forKey:@"name"];//一般key和属性名是取一样的
  32. [aCoder encodeInteger:_age forKey:@"age"];
  33. [aCoder encodeObject:_apples forKey:@"apples"];
  34. }
  35. - (NSString *)description{
  36. NSString *string = [NSString stringWithFormat:@"name=%@,age=%d,apples=%@",_name,_age,_apples];
  37. return string;
  38. }
  39. @end

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

initWithCoder

encodeWithCoder

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

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

  1. - (id)initWithCoder:(NSCoder *)aDecoder{
  2. NSLog(@"initWithCoder");
  3. self = [super init];
  4. if(self != nil){
  5. /*
  6. _name = [aDecoder decodeObjectForKey:@"name"];
  7. _age = [aDecoder decodeObjectForKey:@"age"];
  8. _apples = [aDecoder decodeObjectForKey:@"apples"];
  9. */
  10. //一般我们将key定义成宏,这样就不会出错
  11. _name = [[aDecoder decodeObjectForKey:@"name"] copy];
  12. self.age = [aDecoder decodeObjectForKey:@"age"];
  13. self.apples = [aDecoder decodeObjectForKey:@"apples"];
  14. }
  15. return self;
  16. }

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

其实很简单,就是对属性重新写一下值,然后对每个属性指定一个key就可以了。这个有点类似于Android中的Parcel

(这里我们看到,在解档name属性的时候,用到了copy的一个方法,这个在后面会说到,有浅拷贝和深拷贝之分)

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

  1. //归档时调用此方法
  2. - (void)encodeWithCoder:(NSCoder *)aCoder{
  3. NSLog(@"encodeWithCoder");
  4. [aCoder encodeObject:_name forKey:@"name"];//一般key和属性名是取一样的
  5. [aCoder encodeInteger:_age forKey:@"age"];
  6. [aCoder encodeObject:_apples forKey:@"apples"];
  7. }

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

3、重写description方法

  1. - (NSString *)description{
  2. NSString *string = [NSString stringWithFormat:@"name=%@,age=%d,apples=%@",_name,_age,_apples];
  3. return string;
  4. }

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

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

main.m

  1. //
  2. //  main.m
  3. //  34_ArchiveProtocol
  4. //
  5. //  Created by jiangwei on 14-10-13.
  6. //  Copyright (c) 2014年 jiangwei. All rights reserved.
  7. //
  8. #import <Foundation/Foundation.h>
  9. #import "Person.h"
  10. int main(int argc, const charchar * argv[]) {
  11. @autoreleasepool {
  12. Person *p = [[Person alloc] init];
  13. p.name = @"张三";
  14. p.age = 20;
  15. p.apples = @[@"iphone",@"ipad"];
  16. //归档
  17. NSString *filePath = [NSHomeDirectory() stringByAppendingPathComponent:@"person.archiver"];
  18. BOOL success = [NSKeyedArchiver archiveRootObject:p toFile:filePath];
  19. if(success){
  20. NSLog(@"归档成功");
  21. }
  22. //解归档
  23. Person *person = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
  24. NSLog(@"%@",person);
  25. }
  26. return 0;
  27. }

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

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

总结

这一篇文章我们就说了OC中的归档和解档的相关概念和操作,其实说白了就是将对象写入到文件,和从文件中读取对象。

OC学习篇之---归档和解挡的更多相关文章

  1. (转载)OC学习篇之---归档和解挡

    前几篇文章说到了OC中的Foundation框架,今天我们来看一下OC中的一个重要知识点:归档 OC中的归档就是将对象写入到一个文件中,Java中的ObjectInputStream和ObjectOu ...

  2. OC学习16——对象归档

    转载自  OC学习篇之---归档和解挡 OC中的归档就是将对象写入到一个文件中,Java中的ObjectInputStream和ObjectOutputStream来进行操作的.当然在操作的这些对象都 ...

  3. OC学习篇之---总结和学习目录

    今天终于把OC的基础知识学习完了,但是这些知识只是最基础的,还有很多高级知识,这个可能需要后面慢慢的去学习才能体会到.下面就是这次学习OC的目录教程,如果大家发现有什么不正确的地方,请指正,小弟是新生 ...

  4. OC学习篇之---单例模式

    在之前的一片文章中介绍了对象的拷贝相关知识:http://blog.csdn.net/jiangwei0910410003/article/details/41926531,今天我们来看一下OC中的单 ...

  5. OC学习篇之---文件的操作

    今天我们来介绍OC中文件操作,在之前的文章中,已经接触到了文件的创建了,但是那不是很具体和详细,这篇文章我们就来仔细看一下OC中是如何操作文件的: 第一.首先来看一下本身NSString类给我们提供了 ...

  6. (转载)OC学习篇之---概述

    前言 终于开启了OC的学习篇了,之前由于工作上的事,学习就一直搁浅了,不过最近由于各种原因,感觉必须要开启iOS的开发旅程了,不然就老了.因为之前一直是做Android的,所以学习iOS来就没那么费劲 ...

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

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

  8. (转载)OC学习篇之---Foundation框架中的其他类(NSNumber,NSDate,NSExcetion)

    前一篇说到了Foundation框架中的NSDirctionary类,这一一篇来看一下Foundation的其他常用的类:NSNumber,NSDate,NSException. 注:其实按照Java ...

  9. (转载)OC学习篇之---类的三大特性:封装,继承,多态

    之前的一片文章介绍了OC中类的初始化方法和点语法的使用,今天来继续学习OC中的类的三大特性,我们在学习Java的时候都知道,类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今 ...

随机推荐

  1. js---08函数 定时器

    <!DOCTYPE HTML> <html> <head> <meta http-equiv="Content-Type" content ...

  2. FocusChange-焦点变化监听事件

    想要监听一个控件的焦点变化情况,发现了一个 view.setOnFocusChangeListener(new OnFocusChangeListener() { ...... } 现在写一个小dem ...

  3. View State

    如何查看viewstate 鼠标右键页面,然后view page source 源码中搜索viewstate,会找到一个隐藏的字段. <input type="hidden" ...

  4. tomcat指定配置文件路径方法

    1.在catalina.sh 中设置JAVA_OPTS,例如: JAVA_OPTS='-server -Xms1024m -Xmx1024m -XX:NewSize=128m -XX:MaxPermS ...

  5. c# List集合学习

    1---集合,可以理解成容器 泛型集合 非泛型集合2---使用集合用到的命名空间 using System.Collections.Generic;3---集合是如何来的?集合的前辈是数组,数组在内存 ...

  6. 2017国家集训队作业[agc008f]Black Radius

    2017国家集训队作业[agc008f]Black Radius 时隔4个月,经历了省赛打酱油和中考各种被吊打后,我终于回想起了我博客园的密码= = 题意: ​ 给你一棵树,树上有若干个关键点.选中某 ...

  7. Python day3 知识回顾

    names = ["Jonathen","Joesph","Jotaro","Josuke",["Mea&qu ...

  8. 从”茄子快传”看应用程序怎样获取手机已安装程序的apk文件

    "茄子快传"是联想开发的一款近距离文件共享软件.它通过wifi-direct(速度飞快,不须要联网)或者普通的网络(速度慢)在不同手机间传递文件. 不知为何.它就火了起来,火的也飞 ...

  9. checkbox-padding 调整checkbox字体跟图标距离

    有时候我们会遇到需要调整控件中的内容相对于容器的位置.这里有两种情况 1.linearlayout这样的容器中,包含button类的控件,这时候margin可以调节 2.textview中的文字内容 ...

  10. Android学习笔记进阶十三获得本地全部照片

    这是Intent的一个用法. 在ActivityAction里面有一个“ACTION_GET_CONTENT”字符串常量,该常量让用户选择特定类型的数据. intent.setType("i ...