iOS中经常使用的持久化存储方式有好几种:数据库
偏好设置(NSUserDefaults)数组
plist文件存储缓存
归档安全
SQLite3网络
Core Dataapp
这里不细讲数据库,只针对性地讲讲文件存储、归档/解档、偏好设置等。dom
在此以前,咱们须要先讲讲沙盒(Sandbox)才能继续讲解。学习
每一个iOS应用都有本身的应用沙盒(应用沙盒就是文件系统目录),与其余文件系统隔离。应用必须待在本身的沙盒里,其余应用不能访问该沙盒。沙盒下的目录以下:atom
Application:存放程序源文件,上架前通过数字签名,上架后不可修改spa
Documents: 保存应⽤运行时生成的须要持久化的数据,iTunes同步设备时会备份该目录。例如,游戏应用可将游戏存档保存在该目录
tmp: 保存应⽤运行时所需的临时数据,使⽤完毕后再将相应的文件从该目录删除。应用 没有运行时,系统也可能会清除该目录下的文件。iTunes同步设备时不会备份该目录。
Library/Caches: 保存应用运行时⽣成的须要持久化的数据,iTunes同步设备时不会备份 该目录。⼀通常存储体积大、不须要备份的非重要数据,好比网络数据缓存存储到Caches下
Library/Preference: 保存应用的全部偏好设置,如iOS的Settings(设置) 应⽤会在该目录中查找应⽤的设置信息。iTunes同步设备时会备份该目录
NSUserDefaults是个单例类,用于存储少许数据。NSUserDefaults实际上对plist文件操做的封装,更方便咱们直接操做,通常用于存储系统级别的偏好设置。好比咱们常常将登陆后的用户的一些设置经过NSUserDefaults存储到plist文件中。
有不少App,他们也是将用户的帐号和密码存储在偏好设置中。咱们不讲安全性问题,所以不讨论存储在偏好设置下是否安全。
使用起来很是简单,以下:
1 2 3 4 5 6 7 8 9 10 11 12 |
// 写入文件 - (void)saveUserName:(NSString *)userName password:(NSString *)password { [[NSUserDefaults standardUserDefaults] setObject:userName forKey:@"username"]; [[NSUserDefaults standardUserDefaults] setObject:password forKey:@"password"]; [[NSUserDefaults standardUserDefaults] synchronize]; }
// 在用的时候,就能够读取出来使用 NSString *userName = [[NSUserDefaults standardUserDefaults] objectForKey:@"username"]; NSString *password = [[NSUserDefaults standardUserDefaults] objectForKey:@"password"];
|
存储到偏好设置的只有系统已经提供好的类型,好比基本类型、NSNumber、NSDictionary、NSArray等。对于NSObject及继承于NSObject的类型,是不支持的。以下:
1 2 3 4 5 6 7 |
NSObject *obj = [[NSObject alloc] init]; [[NSUserDefaults standardUserDefaults] setObject:obj forKey:@"obj"];
// 就会崩溃 Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: 'Attempt to insert non-property list object <NSObject: 0x7fb502680cb0> for key obj'
|
有的时候,咱们须要将下载的数据存储到文件中存储起来,好比,有时候咱们将下载起来的城市的数据存储到本地,当更新城市的顺序时,下次也可以按照最后一次操做的顺序来显示出来。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
// 数据存储,是保存到手机里面, // Plist存储,就是把某些数据写到plist文件中 // plist存储通常用来存储数组和字典 // Plist存储是苹果特有,只有苹果才能生成plist // plist不能存储自定义对象,如NSObject、model等 NSDictionary *dict = @{@"age":@"18",@"name":@"USER"};
// 保存应用沙盒(app安装到手机上的文件夹) // Caches文件夹 // 在某个范围内容搜索文件夹的路径 // directory:获取哪一个文件夹 // domainMask:在哪一个范围下获取 NSUserDomainMask:在用户的范围内搜索 // expandTilde是否展开全路径,YES:展开 NSString *cachePath = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)[0]; NSLog(@"%@",cachePath);
// 拼接文件路径 NSString *filePath = [cachePath stringByAppendingPathComponent:@"data.plist"];
// 获取应用沙盒 NSString *homePath = NSHomeDirectory(); NSLog(@"%@",homePath);
// File:文件全路径 => 全部文件夹路径 + 文件路径 [dict writeToFile:filePath atomically:YES]; // 将数据取出来 NSLog(@"%@", [NSDictionary dictionaryWithContentsOfFile:filePath]);
|
咱们看看打印的结果:
1 2 3 4 5 6 7 8 |
2016-02-17 22:14:43.055 iOSPersistentStorageDemo[25471:809758] /Users/huangyibiao/Library/Developer/CoreSimulator/Devices/CF3A5A4C-486F-4A72-957B-2AD94BD90EC1/data/Containers/Data/Application/65E8F814-45E5-420C-A174-822A7830748E/Library/Caches 2016-02-17 22:14:43.055 iOSPersistentStorageDemo[25471:809758] /Users/huangyibiao/Library/Developer/CoreSimulator/Devices/CF3A5A4C-486F-4A72-957B-2AD94BD90EC1/data/Containers/Data/Application/65E8F814-45E5-420C-A174-822A7830748E 2016-02-17 22:14:43.056 iOSPersistentStorageDemo[25471:809758] { age = 18; name = USER; }
|
注意:操做plist文件时,文件路径必定要是全路径。
自定义对象应用范围很广,由于它对应着MVC中的Model层,即实体类。在程序中,咱们会在Model层定义不少的entity,例如User、Teacher、Person等。
那么对自定义对象的归档显得重要的多,由于不少状况下咱们须要在Home键以后保存数据,在程序恢复时从新加载,那么,归档即是一个好的选择。
下面咱们自定义一个Person类:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 |
// 要使对象能够归档,必须遵照NSCoding协议 @interface Person : NSObject<NSCoding>
@property (nonatomic, assign) int age; @property (nonatomic, strong) NSString *name;
@end
@implementation Person
// 何时调用:只要一个自定义对象归档的时候就会调用 - (void)encodeWithCoder:(NSCoder *)aCoder { [aCoder encodeObject:self.name forKey:@"name"]; [aCoder encodeInt:self.age forKey:@"age"]; }
- (id)initWithCoder:(NSCoder *)aDecoder { if (self = [super init]) { self.name = [aDecoder decodeObjectForKey:@"name"]; self.age = [aDecoder decodeIntForKey:@"age"]; } return self; } @end
|
如何将自定义对象归档和解档:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
- (void)savePerson { // 归档:plist存储不能存储自定义对象,此时能够使用归档来完成 Person *person = [[Person alloc] init]; person.age = 18; person.name = @"USER";
// 获取tmp目录路径 NSString *tempPath = NSTemporaryDirectory();
// 拼接文件名 NSString *filePath = [tempPath stringByAppendingPathComponent:@"person.data"];
// 归档 [NSKeyedArchiver archiveRootObject:person toFile:filePath]; }
- (void)readPerson { // 获取tmp NSString *tempPath = NSTemporaryDirectory();
// 拼接文件名 NSString *filePath = [tempPath stringByAppendingPathComponent:@"person.data"];
// 解档 Person *p = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath]; NSLog(@"%@ %d",p.name,p.age); }
|
假设咱们定义了一个自定义的view,这个view是用xib或者storybard来生成的,那么咱们我必定以下方法时,就须要以下实现:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
@implementation CustomView
// 解析xib,storyboard文件时会调用 - (id)initWithCoder:(NSCoder *)aDecoder { // 何时调用[super initWithCoder:aDecoder]? // 只要父类遵照了NSCoding协议,就调用[super initWithCoder:aDecoder] if (self = [super initWithCoder:aDecoder]) { NSLog(@"%s",__func__); }
return self; }
@end
|
若是想要学习如何经过runtime来实现自动归档、解档,请阅读文章:经过runtime自动归档/解档