一、NSBundle资源包。

只要把文件拖到Xcode左边项目导航面板中,选择复制文件到项目中,该文件就包含进bundle中了。用[NSBundle mainBundle]获取应用程序包,常用的方法:

  • URLForResource:withExtension:根据资源名和扩展名获取对应的URL。
  • pathForResource:ofType:根据资源名和类型名获取对应的路径。
  • resourcePath:返回该NSBundle的子目录所包含资源的完整路径。

二、NSKeyedArchiver 、NSKeyedUnarchiver归档和恢复。

归档就是用某种格式把一个或多个对象保存到指定的文件中(把对象转化为可保存、可传输的数据流),在需要的时候再从文件中恢复它们(从数据流中恢复该对象)。

NSKeyedArchiver和NSKeyedUnarchiver继承自NSCoder。

1、NSKeyedArchiver的简单用法(适用于非自定义OC对象)。

  • 直接调用NSKeyedArchiver的archiveRootObject:toFile:类方法,指定一个对象为root进行归档。

     NSDictionary *dict = @{@"xiaohong" : @"小红", @"xiaoming" : @"小明"};
    NSString *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).lastObject;
    path = [path stringByAppendingPathComponent:@"dict.arch"];
    BOOL isSucc = [NSKeyedArchiver archiveRootObject:dict toFile:path];
  • 或者调用NSKeyedArchiver的archivedDataWithRootObject:类方法,把一个对象转换成NSData的类型并返回,再写入本地或网络。
     NSDictionary *dict = @{@"xiaohong" : @"小红", @"xiaoming" : @"小明"};
    NSData *data = [NSKeyedArchiver archivedDataWithRootObject:dict];
    NSString *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).lastObject;
    path = [path stringByAppendingPathComponent:@"data.arch"];
    [data writeToFile:path atomically:YES];
  • 直接调用NSKeyedUnarchiver的unarchiveObjectWithFile:类方法进行恢复。
  •  NSString *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).lastObject;
    path = [path stringByAppendingPathComponent:@"dict.arch"];
    NSDictionary *dict = [NSKeyedUnarchiver unarchiveObjectWithFile:path];
  • 或者调用NSKeyedUnarchiver的unarchiveObjectWithData:类方法,对已经归档成NSData的单对象进行恢复。
     NSString *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).lastObject;
    path = [path stringByAppendingPathComponent:@"data1.arch"];
    NSData *data = [NSData dataWithContentsOfFile:path];
    NSDictionary *dicts = [NSKeyedUnarchiver unarchiveObjectWithData:data];

2、NSCoding协议的用法(适用于自定义OC对象)。

如果程序需要归档或恢复任意自定义对象,则该类必需实现NSCoding协议,并实现协议中定义的方法。

  •  #import <UIKit/UIKit.h>
    
     @interface HLPerson : NSObject <NSCoding>
    
     @property (copy, nonatomic) NSString *name;
    @property (assign, nonatomic) CGFloat height; @end
  • 归档该对象时,总会调用encodeWithCoder:方法,在此方法中分别调用NSCoder的encodeXxx:forKey:方法归档对应的成员变量。
  • 回复该对象时,总会调用initWithCoder:方法,在此方法中分别调用NSCoder的decodeXxxForKey:方法恢复对应的成员变量的值。
     #import "HLPerson.h"
    
     @implementation HLPerson
    
     - (void)encodeWithCoder:(NSCoder *)aCoder {
    [aCoder encodeObject:self.name forKey:@"name"];
    [aCoder encodeFloat:self.height forKey:@"height"];
    } - (instancetype)initWithCoder:(NSCoder *)aDecoder {
    self.name = [aDecoder decodeObjectForKey:@"name"];
    self.height = [aDecoder decodeFloatForKey:@"height"];
    return self;
    } @end

然后就可以像归档普通对象一样归档我们自定义的HLPerson对象了,参看1。

3、用NSData自定义归档。

前面的方法只能归档或恢复单个的OC对象,若要归档或恢复多个对象到单个文件中,则可以借助NSMutableData来创建NSKeyedArchiver或者NsKeyedUnarchiver对象。

(1)、归档步骤。

  • 创建一个NSMutableData对象,并以此对象为参数调用initForWritingWithMutableData创建NSKeyedArchiver对象。
  • 重复调用NSKeyedArchiver对象的encodeXxx:forKey:方法来归档所有需要归档到一个文件中的对象。
  • 调用NSKeyedArchiver对象的finishEncoding方法结束归档。
  • 保存该NSMutableData对象到本地或网络。
     HLPerson *person = [[HLPerson alloc] init]; // 自定义的对象需实现NSCoding协议
    person.name = @"hahh";
    person.height = ; NSDictionary *dict = @{@"xiaohong" : @"小红", @"xiaoming" : @"小明"};
    NSArray *array = @[person, dict];
    CGFloat age = 12.5; NSMutableData *data = [NSMutableData data];
    NSKeyedArchiver *arch = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
    [arch encodeObject:person forKey:@"person"];
    [arch encodeObject:dict forKey:@"dict"];
    [arch encodeObject:array forKey:@"array"];
    [arch encodeFloat:age forKey:@"age"];
    [arch finishEncoding]; NSString *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).lastObject;
    path = [path stringByAppendingPathComponent:@"data.arch"];
    BOOL isSucc = [data writeToFile:path atomically:YES];

(2)、恢复步骤。

  • 以NSData为参数创建NSKeyedUnarchiver对象。
  • 重复调用NSKeyedUnarchiver对象的decodeXxx:forKey:方法来从一个文件中恢复所有归档过的对象。
  • 调用NSKeyedUnarchiver对象的finishEncoding方法结束恢复。
     NSString *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).lastObject;
    path = [path stringByAppendingPathComponent:@"data.arch"];
    NSData *data = [NSData dataWithContentsOfFile:path];
    NSKeyedUnarchiver *unarch = [[NSKeyedUnarchiver alloc] initForReadingWithData:data]; HLPerson *person = [unarch decodeObjectForKey:@"person"];
    NSDictionary *dict = [unarch decodeObjectForKey:@"dict"];
    NSArray *array = [unarch decodeObjectForKey:@"array"];
    CGFloat age = [unarch decodeFloatForKey:@"age"]; [unarch finishDecoding];

延伸:归档会将整个对象转换成字节数据,包括该对象的所有成员变量,如果该成员变量指向另一个OC对象,则此OC对象也将一起被归档成字节数据。这就表明:当归档一个对象时,系统会把该对象关联的所有数据,都转换为字节数据。如果程序从这些字节数据中恢复对象,则恢复出来的对象和原来的对象相同,但在内存上的地址不同,这就实现了对象的深复制。

三、NSUserDefaults

NSUserDefaults是一个单例类,一般用来保存程序参数、用户账号、选项相关的少量数据,用它存储的对象放在沙盒的Library/Preferences目录下的文件中。

1、存储OC类型的对象:

  • standardUserDefaults:获取该单例对象。
  • setXxx:forKey:根据key设置要存储的对象。

  • synchronize:保存对象到Library/Preferences目录下的文件中。

  • xxxForKey:根据key获取对应的对象。
     NSDictionary *dict1 = @{@"xiaohong" : @"小红", @"xiaoming" : @"小明"};
    CGFloat height = 111.1;
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    [defaults setObject:dict1 forKey:@"dict1"];
    [defaults setFloat:height forKey:@"height"];
    [defaults synchronize];
    NSDictionary *dict2 = [defaults objectForKey:@"dict1"];
    NSLog(@"%@ ---- %@ ---- %.2f", dict2[@"xiaohong"], dict2[@"xiaoming"], height);

NSUserDefaults支持的数据类型有:NSNumber(NSInteger、float、double),NSString,NSDate,NSArray,NSDictionary,NSData,BOOL。存储的对象全是不可变的。

2、存储自定义类型的对象:

NSUserDefaults不支持自定义对象,可以把自定义对象转成NSData类型进行存储:

  • 自定义的类需遵守NSCoding协议,且实现协议方法:encodeWithCoder:和initWithCoder:。
  • 第一种方法是:调用NSKeyedArchiver的archivedDataWithRootObject:类方法,把一个对象转换成NSData的类型并返回。第二中方法是:创建一个NSMutableData对象,并以此对象为参数调用initForWritingWithMutableData创建NSKeyedArchiver对象,重复调用NSKeyedArchiver对象的decodeObjectForKey:forKey:方法来归档所有需要归档到一个文件中的对象,调用NSKeyedArchiver对象的finishEncoding方法结束归档。
  • 调用NSUserDefaults的setObject:dict1 forKey:存储上一步骤得到的NSData或NSMutableData对象。(参考本篇二)

恢复对象:

  • 调用NSUserDefaults的objectForKey:取出对应的NSData对象。
  • 第一种方法是:调用NSKeyedUnarchiver的unarchiveObjectWithData:类方法,对已经归档成NSData的对象进行恢复。第二中方法是:以上一步得到的NSData对象为参数创建NSKeyedUnarchiver对象,重复调用NSKeyedUnarchiver对象的decodeObject:forKey:方法来从一个文件中恢复所有归档过的对象,调用NSKeyedUnarchiver对象的finishEncoding方法结束恢复。(参考本篇二)

注:对相同的key进行赋值,相当于覆盖原有的值。

四、属性列表

属性列表适合保存少量简单的数据,Xcode能以图形化的方式来编辑属性列表文件。

  • writeToFile:atomically:把目标对象写入文件。
  • xxxWithContentsOfFile:初始化对象。
     NSDictionary *dict1 = @{@"xiaohong" : @"小红", @"xiaoming" : @"小明"};
    NSDictionary *dict2 = @{@"xiaogong" : @"小刚", @"xiaoqiang" : @"小强"};
    NSArray *array1 = @[dict1, dict2]; NSString *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).lastObject;
    path = [path stringByAppendingPathComponent:@"data.plist"];
    [array1 writeToFile:path atomically:YES]; NSArray *array2 = [NSArray arrayWithContentsOfFile:path];
    NSLog(@"%@ ---- %@ ---- %@ ---- %@", [array2[] objectForKey:@"xiaohong"], [array2[] objectForKey:@"xiaoming"], [array2[] objectForKey:@"xiaogong"] ,[array2[] objectForKey:@"xiaoqiang"]);

属性列表只能保存下列类型的对象:NSArray、NSMutableArray、NSDictionary、NSMutableDictionary、NSData、NSMutableData、NSString、NSMutableString、NSValue和NSNumber。

如果需要保存自定义的对象,可以使用对象归档的方式,详见二。

五、SQLite3

详见SQLite3的基本使用

六、Core Data 

详见Core Data的一些常见用法

七、FMDB

详见FMDB的简单使用

 

数据存储与IO(二)的更多相关文章

  1. 数据存储与IO(一)

    应用程序沙盒简介:iOS应用程序只能在系统为它分配的文件区域内读写文件,这个区域就是此应用程序的沙盒,Application目录下的GUID文件夹就是沙盒,这个文件夹是系统随机命名的.程序所有的非代码 ...

  2. android学习笔记45——android的数据存储和IO

    android的数据存储和IO SharedPreferences与Editor简介 SharedPreferences保存的数据主要是类似于配置信息格式的数据,因此其保存的数据主要是简单的类型的ke ...

  3. Android系统的五种数据存储形式(二)

    之前介绍了Android系统下三种数据存储形式,今天补充介绍另外两种,分别是内容提供者和网络存储.有些人可能认为内存提供者和网络存储更偏向于对数据的操作而不是数据的存储,但这两种方式确实与数据有关,所 ...

  4. 【Android】数据存储-java IO流文件存储

    1.数据持久化:将在内存中的瞬时数据保存在存储设备中.瞬时数据:设备关机数据丢失.持久化技术提供一种机制可以让数据在瞬时状态和持久状态之间转换. 2.Android中简单的三种存储方式:文件存储.Sh ...

  5. Android数据存储之IO

    Android开发中免不了数据本地的存储,今天我们来说一说怎样利用IO流来进行数据存储. 这里我们通过模拟一个QQ登陆界面的小demo来实际操作IO流. 功能描写叙述:点击button能够保存用户输入 ...

  6. Spark RDD概念学习系列之Spark的数据存储(十二)

    Spark数据存储的核心是弹性分布式数据集(RDD). RDD可以被抽象地理解为一个大的数组(Array),但是这个数组是分布在集群上的. 逻辑上RDD的每个分区叫一个Partition. 在Spar ...

  7. IOS高级编程之二:IOS的数据存储与IO

    一.应用程序沙盒 IOS应用程序职能在系统为该应用所分配的文件区域下读写文件,这个文件区域就是应用程序沙盒.所有的非代码文件如:图片.声音.映象等等都存放在此. 在mac中command+shift+ ...

  8. [ Android 五种数据存储方式之二 ] —— 文件存储数据

    关于文件存储,Activity提供了openFileOutput()方法可以用于把数据输出到文件中,具体的实现过程与在J2SE环境中保存数据到文件中是一样的. 文件可用来存放大量数据,如文本.图片.音 ...

  9. edgedb 内部pg 数据存储的探索 (二) 创建数据库命令说明

    前面已经创建好了一个简单可以访问pg 的edgedb 环境,现在测试几个数据库操作命令在pg 的存储 创建数据库 连接环境 注意账户是按照上次创建的环境配置 edgedb -u edgedb 创建数据 ...

随机推荐

  1. 原生js实现简洁的返回顶部组件

    本文内容相当简单,所以没有发布到博客园首页,如果你不幸看到,那只能是我这篇文章的荣幸,谢谢你的大驾光临~(本博客返回顶部的功能就使用的是这个组件) 返回顶部组件是一种极其常见的网页功能,需求简单:页面 ...

  2. MSSQLLocalDB 连接字符串 vs2015

    <add key="MSConnectionString" value="Data Source=(localdb)\MSSQLLocalDB;Initial Ca ...

  3. Android的px、dp和sp

    Android的px.dp和sppx: 即像素,1px代表屏幕上一个物理的像素点:偶尔用到px的情况,是需要画1像素表格线或阴影线的时候. dp: 这个是最常用但也最难理解的尺寸单位.它与“像素密度” ...

  4. webstorage[html5的本地数据处理]

    1.webStorage是什么? webStorage是html5中用于本地化存储的一种方式,而在之前呢我们是用cookie的存储方式处理; 2.那它们之间的区别是什么? Ⅰ.cookie存在的问题: ...

  5. C#使用Process调用批处理阻塞问题

    PS:又见到熟悉的C#代码了,好开心,哈哈哈.这次又跳坑了,好不容易才爬起来.   公司有自己开发的一套Submit.Compile.Publish的生命周期系统. 在Compile时,需要调用外部的 ...

  6. VS2010如何使用Visual Studio Online在线服务管理团队资源(在线TFS)

    前言 Visual Studio Online,也就是以前的Team Foundation Service,从名字可以看出这是一个团队资源管理服务.在微软的云基础架构中运行,无需安装或配置任何服务器, ...

  7. 调优Java virtual machine常见问题汇总整理

    数据类型 Java虚拟机中,数据类型可以分为两类:基本类型和引用类型.基本类型的变量保存原始值,即:他代表的值就是数值本身:而引用类型的变量保存引用值.“引用值”代表了某个对象的引用,而不是对象本身, ...

  8. Python私有函数和公开函数

    类似_xxx和__xxx这样的函数或变量就是非公开的(private),不应该被直接引用,比如_abc,__abc等: # private私有函数 def _private_1(name): retu ...

  9. mysql memory表性能测试以及使用场景

    最近公司计划将风控逻辑移到slave库进行计算,因为考虑到业务表数据会比较大,此时如果还是走nest-loop的话,即使unique进行连接,因为还是需要至少2次以上LIO才能读一条记录,如果达到类似 ...

  10. Kickoff - 创造可扩展的,响应式的网站

    Kickoff 是一个轻量级的前端框架,用于创建可扩展的,响应式的网站.作为前端开发人员,我们工作的类型越来越多样化.Kickoff 旨在帮助您在所有项目保持一致的结构和风格,无需添加其他框架. 在线 ...