曾经接触过iOS开发,并且开发过两个应用,纵然青涩,也算是一断美好的回忆。转眼就已经一年多了!现在回过头来决定再次拿起iOS开发。

下面讲NSData:

NSdata的概念

1、使用文件时需要频繁地将数据读入一个临时存储区,它通常称为缓冲区

2、NSdata类提供了一种简单的方式,它用来设置缓冲区,将文件的内容读入缓冲区,或者将缓冲区内容写到一个文件。

3、对于32位应用程序,NSdata缓存最多2GB

4、我们有两种定义 NSData(不可变缓冲区),NSMutableData(可变缓冲区)

上面的描述有点片面,再补充一些:

NSData就是字节流的数据,它为字节流提供面向对象的存储空间,能够把一些字符串或图片等等一些非常复杂的数据类型转换成01字节流。在Objective-c中几乎所有的对象类型都可以转换成NSData(字符串也属于对象类型),基本数据类型需要先封装成对象,再转换成NSData。

在iOS中将对象序列化时,就需要将对象转换成为字节流数据即NSData类型的数据。

在要求将数据进行网络传输的场合,也需要将数据转化成为字节流才能进行传输。无论是向服务器写数据、还是从服务器返回的数据都应该是NSData类型的。

字定义保存内存中对象的方法

数据流图如下:

Object  <----->  NSKeyedArchiver/NSKeyedUnarchiver  <------>  NSData <-----> NSString( file's name )

在Objective-c中要将对象转换成为NSData类型时,对象的类必须遵守NSCoding协议。NSCoding协议规定了两个必须实现的方法:initWithCoder:和encodeWithCoder:。initWithCoder:方法使用给定的状态初始化一个新对象,也称为反序列化,或者解码。encodeWithCoder:方法接收一个带状态的对象,并对其进行序列化,或者称为编码。您可以把encodeWithCoder:方法理解成将对象打包准备传输,把initWithCoder:方法理解成将接收到的数据解包成可用的对象。

许多常用的类,例如NSString、NSArray、NSNumber以及许多其他的类,已经实现了NSCoding协议。对于自定义的类,就需要用户自己实现NSCoding协议的initWithCoder:方法和encodeWithCoder:方法。这并不是很麻烦的事,可以看到一个实现NSCoding协议的例子:

@interface BigData : NSObject <NSCoding>
#pragma mark NSCoding
#define kTitleKey @"Title"
#define kRatingKey @"Rating" - (void) encodeWithCoder:(NSCoder *)encoder {
[encoder encodeObject:_title forKey:kTitleKey];
[encoder encodeFloat:_rating forKey:kRatingKey];
} - (id)initWithCoder:(NSCoder *)decoder { NSString *title = [decoder decodeObjectForKey:kTitleKey];
float rating = [decoder decodeFloatForKey:kRatingKey];
return [self initWithTitle:title rating:rating]; }

在encodeWithCoder中,我们传入一个NSCoder对象,通过helper 方法把它编码成细小的数据片。这些helper方法有: encodeObject,encodeFloat,encodeInt等等。

在每一次encode的时候需要提供一个key用于以后decode的时候查找。通常,我们会对这些类加一个field,减一个field。为了让你的程序更健壮,在你decode一个field的时候,最好判断一下它的值是不是nil或者零,然后给它赋一个合适的默认值。

操作文件结构并保存

上代码,一个.m源文件:

#import "DataModel.h"
#import "NotePage.h" @implementation DataModel -(DataModel *)init{ [self loadPages];
return self;
} #pragma mark file storage
-(NSString*)documentsDirectory{ NSArray *paths =
NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,
NSUserDomainMask, YES);
NSString *documentsDirectory = [paths firstObject];
return documentsDirectory;
} -(NSString*)filePathOfPage:(NotePage *)page{ return [[self documentsDirectory]
stringByAppendingPathComponent:[NSString stringWithFormat:@"page%d.plist",page.pageID]]; } -(void)savePage:(NotePage *)page{ NSMutableData *data = [[NSMutableData alloc]init];
NSKeyedArchiver *archiver = [[NSKeyedArchiver
alloc]initForWritingWithMutableData:data];
[archiver encodeObject:page forKey:[NSString stringWithFormat:@"page%d.plist",page.pageID] ];
[archiver finishEncoding];
[data writeToFile:[self filePathOfPage:page] atomically:YES];
} -(void)removePage:(NotePage *)page{ NSFileManager *fileManager = [NSFileManager defaultManager];
NSString *path = [self filePathOfPage:page ];
NSError * error;
[fileManager removeItemAtPath:path error:&error];
int index = [self.pages indexOfObject:page];
[self.pages removeObjectAtIndex:index]; } -(void)loadPages{ self.pages = [[NSMutableArray alloc]initWithCapacity:];
NSFileManager *fileManager = [NSFileManager defaultManager];
NSString *documentsPath = [self documentsDirectory];
NSError *error;
NSArray *fileList = [fileManager contentsOfDirectoryAtPath:documentsPath error:&error]; NSString *path;
NSData * data;
NSKeyedUnarchiver *unarchiver;
for (int i = ; i< fileList.count ; i++) {
if([[fileList[i] substringWithRange:NSMakeRange(, )] isEqualToString:@"page"]){
path = [documentsPath stringByAppendingPathComponent:fileList[i]];
data = [[NSData alloc]initWithContentsOfFile:path];
unarchiver = [[NSKeyedUnarchiver alloc]initForReadingWithData:data];
NotePage *page = [unarchiver decodeObjectForKey:fileList[i]];
[self.pages addObject:page];
}
}
} @end

保存对象:

进程执行一个archiver对象中的方法,把对象编码成字节流保存到一个data对象中;然后执行data对象中的方法,把它保存的字节流写到一个文件中。

取出对象:

进程执行一个data对象中的方法,把文件中的字节流读出来;然后执行一个unarchiver对象中的方法,把data对象中的字节流解码还原成对象。

注意,编码和解码方法在要保存的对象内。

参考上文中的数据流图。

JAVA中的文件操作

数据流图如下:

BufferedReader <-------> InputStreamReader <-------> FileInputStream <------> File ( file's pointer )

ps:

另外博主找了一篇文章,算是概述的形式回顾一下iPhone开发吧! http://www.cnblogs.com/jy578154186/archive/2013/02/27/2934881.html

用NSData和NSFileManager保存内存中的对象的更多相关文章

  1. JavaScript 变量类型 保存内存中的位置 和 引用

    1. JavaScript变量 基本类型值在内存中占据固定大小的空间 因此被保存在栈内存中. 从一个变量向另一个变量复制基本来下的值 会创建这个值得一个副本. 引用类型的值是对象 保存在堆内存中. 包 ...

  2. 浅谈Java虚拟机内存中的对象创建,内存布局,访问定位

    参考于 深入理解Java虚拟机 这里介绍HotSpot虚拟机(自带的虚拟机) 1.对象的创建 对于程序员来说,创建对象的方法: User user1 = new User(); User user2 ...

  3. java内存中的对象

    前记:几天前,在浏览网页时偶然的发现一道以前就看过很多遍的面试题,题目是:“请说出‘equals’和‘==’的区别”,当时我觉得我还是挺懂的,在心里答了一点(比如我们都知道的:‘==’比较两个引用是否 ...

  4. 图解JVM在内存中申请对象及垃圾回收流程

    http://longdick.iteye.com/blog/468368 先看一下JVM的内存模型: 从大的方面来讲,JVM的内存模型分为两大块: 永久区内存( Permanent space )和 ...

  5. linq读书笔记2-查询内存中的对象

    上次我们说到了linq对数组内容的检索,自.net2.0以后,泛型成了很常见的一种应用技术,linq对泛型的检索也提供了完善的支持 如对list类型的支持,范例如下: class Program    ...

  6. 内存中 OLTP - 常见的工作负荷模式和迁移注意事项(二)

    ----------------------------我是分割线------------------------------- 本文翻译自微软白皮书<In-Memory OLTP – Comm ...

  7. PHP对象类型在内存中的分配

    对象类型和整型.字符串等类型一样,也是PHP中的一种数据类型.都是在程序中用于存储不同类型数据使用的,在程序运行时它的每部分内容都要先加载到内存中再被使用.那么对象类型的数据在内存中是如何分配的呢?先 ...

  8. java中的各种数据类型在内存中存储的方式

    原文地址:http://blog.csdn.net/aaa1117a8w5s6d/article/details/8251456 1.Java是如何管理内存的 java的内存管理就是对象的分配和释放问 ...

  9. 详解Python变量在内存中的存储

    这篇文章主要是对python中的数据进行认识,对于很多初学者来讲,其实数据的认识是最重要的,也是最容易出错的.本文结合数据与内存形态讲解python中的数据,内容包括: 引用与对象 可变数据类型与不可 ...

随机推荐

  1. uva 10131

    DP 先对大象体重排序   然后寻找智力的最长升序子列  输出路径.... #include <iostream> #include <cstring> #include &l ...

  2. XEE介绍

    摘要: XMl Entity Expansion(攻击)某种程度上类似于 XML Entity Expansion,但是它主要试图通过消耗目标程序的服务器环境来进行DOS攻击的.这种攻击基于XML E ...

  3. 设计模式(Design Patterns——可复用面向对象软件的基础

        设 计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了可重用代码.让代码更容易被他人理解.保证代 码可靠性. 毫无疑问 ...

  4. 2013 Multi-University Training Contest 1 Cards

    数据不是很大,直接枚举约数,判断4个条件是否满足! 这样就得到4种卡片,总共2^4种情况,枚举各种情况即可!!! #include<iostream> #include<cmath& ...

  5. python之self (转)

    首先明确的是self只有在类的方法中才会有,独立的函数或方法是不必带有self的.self在定义类的方法时是必须有的,虽然在调用时不必传入相应的参数. self名称不是必须的,在python中self ...

  6. Biba模型简介

    上周上信息安全的课,老师留了个Biba模型的作业.自己看书了解了一下,记录如下. 参考资料:石文昌<信息系统安全概论第2版> ISBN:978-7-121-22143-9 Biba模型是毕 ...

  7. lintcode :数组剔除元素后的乘积

    题目: 数组剔除元素后的乘积 给定一个整数数组A. 定义B[i] = A[0] * ... * A[i-1] * A[i+1] * ... * A[n-1], 计算B的时候请不要使用除法. 样例 给出 ...

  8. Tomcat处理HTTP请求源码分析(下)

    转载:http://www.infoq.com/cn/articles/zh-tomcat-http-request-2 很多开源应用服务器都是集成tomcat作为web container的,而且对 ...

  9. @Override在JDK1.5和JDK1.6中用法区别

    @Override 注解在jdk1.5环境下,只能用于对基类(父类)的方法的重写.而不能用于对实现的接口的方法的实现.而在jdk1.6环境下,两者都适用.

  10. 怎样加快master数据库的写操作?分表原则!将表水平划分!或者添加写数据库的集群

    1.怎样加快master数据库的写操作?分表原则!将表水平划分!减少表的锁定时间!!! 或者或者添加写数据库的集群!!!或者添加写数据库的集群!!! 2.既然分表了,就一定要注意分表的规则!要在代码层 ...