1、概念
数据持久化-将数据保存在硬盘中,使得机器/应用程序重启后,可以继续访问之前保存的数据。
2、方案
plist(属性列表)、preference(偏好设置)、NSKeyedArchiver(归档)、SQLite 3(数据库)、CoreData
3、沙盒机制
3.1、应用程序的目录路经
NSString *appPath = NSHomeDirectory();
appPath:/Users/lizhongfeng/Library/Developer/CoreSimulator/Devices/6E35AB76-8538-4BB8-9F2A-1574A807C05B/data/Containers/Data/Application/39D4B302-E8BA-4DC0-A9A2-E47DCCD89615
上面的代码得到的是app的目录路径,即为该app的沙盒路径。
3.2、沙盒的目录结构
虽然沙盒中文件夹较多,但这些文件夹不尽相同,所存放的数据类型不同。因此,存放数据时,应根据数据类型和想要达到的效果,选择合适的文件夹。
3.2.1、Documents
最常用的目录,iTunes同步该应用时,会同步此文件夹中的内容,适合存储重要数据;
NSString *documentsPath = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).firstObject;
documentsPath:/Users/lizhongfeng/Library/Developer/CoreSimulator/Devices/6E35AB76-8538-4BB8-9F2A-1574A807C05B/data/Containers/Data/Application/0967BD85-9B06-449B-A1F0-27B542BAC69A/Documents
3.2.2、Library/Caches
iTunes不会同步此文件夹,适合存储体积大,但无需备份的非重要数据;
NSString *cachesPath = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES).firstObject;
cachesPath:/Users/lizhongfeng/Library/Developer/CoreSimulator/Devices/6E35AB76-8538-4BB8-9F2A-1574A807C05B/data/Containers/Data/Application/0967BD85-9B06-449B-A1F0-27B542BAC69A/Library/Caches
3.2.3、Library/Preferences
iTunes同步该应用时,会同步此文件夹中的内容,通常保存应用的设置信息;
3.2.4、tmp
iTunes不会同步此文件夹,系统可能在应用没运行时,就删除该目录下的文件,所以该目录适合保存应用中的一些临时文件,用完就删除。
NSString *tmpPath = NSTemporaryDirectory();
tmpPath:/Users/lizhongfeng/Library/Developer/CoreSimulator/Devices/6E35AB76-8538-4BB8-9F2A-1574A807C05B/data/Containers/Data/Application/5A198C87-225D-4263-BCDF-9CC28A9DFB4F/tmp/
4、plist文件
plist文件是将某些类,通过XML文件的形式保存在目录中。
4.1、可以被序列化的类型
NSArray/NSMutableArray、NSDictionary/NSMutableDictionary、NSData/NSMutableData、NSString/NSMutableString、NSNumber、NSDate,只有以上类型才能使用plist文件存储。
4.2、存储步骤
// 拼接文件路径
NSString *docPath = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).firstObject;
NSString *filePath = [docPath stringByAppendingPathComponent:@"123.plist"];
// 存储
NSArray *array = @[@"123",@"456",@"zhongfeng"];
[array writeToFile:filePath atomically:YES];
// 读取
NSArray *result = [NSArray arrayWithContentsOfFile:filePath];
NSLog(@"result:%@",result);
注意:
存储时使用writeToFile: atomically:方法。其中,atomically:表示是否先写入一个临时文件,再将该临时文件拷贝到目标文件地址。将其设置为YES是一种更安全的文件写入方法,所以一般都写YES。
读取文件使用***WithContentsOfFile:(arrayWithContentsOfFile:,dictionaryWithContentsOfFile:)方法。
5、Preference
5.1、使用方法
// 获得NSUserDefaults文件
NSUserDefaults *userDefault = [NSUserDefaults standardUserDefaults];
// 项文件中写入内容
[userDefault setObject:@"zhongfeng" forKey:@"name"];
[userDefault setInteger:27 forKey:@"age"];
[userDefault setBool:YES forKey:@"isMan"];
// 同步(手动调用synchronize方法以及时保存userDefault文件内容)
[userDefault synchronize];
// 读取
NSString *name = [userDefault stringForKey:@"name"];
NSInteger age = [userDefault integerForKey:@"age"];
BOOL isMan = [userDefault boolForKey:@"isMan"];
NSLog(@"name = %@, age = %ld, isMan = %d",name,(long)age,isMan);
注意:
偏好设置一般是用来保存应用程序的配置信息,不要在偏好设置中保存其他数据。
如果没有调用synchronize方法进行同步,则NSUserDefaults会过一段时间自动调用synchronize方法进行同步。因此,如果需要立即写入文件就必须马上调用synchronize方法。
偏好设置会将所有数据保存在Preferences中一个以应用程序包名(Bundle Identifier)的plist文件。因此,偏好设置所能存储的数据类型肯定是plist文件所支持的类型。
6、NSKeyedArchiver
归档是iOS中另一种形式化的序列化,只要遵循了NSCoding协议的对象都可以通过它实现序列化。由于绝大数支持存储数据的Foundation和Coca Touch类都遵循了NSCoding协议。因此,对于大多数类而言,归档还是比较容易实现。
6.1、NSCoding
NSCoding协议声明了两个方法,这两个方法都必须要实现。一个方法说明如何将对象编码到归档中;另一个方法说明如何解档获取一个对象。
/*遵循协议、设置属性*/
#import@interface Person : NSObject// 遵循NSCoding协议
// 属性
@property (nonatomic,copy) NSString *name;
@property (nonatomic,assign) NSInteger age;
@property (nonatomic,assign) BOOL isMan;
// 初始化Person
+(instancetype)initPersonWithName:(NSString *)name age:(NSInteger)age isMan:(BOOL)isMan;
@end
/*实现NSCoding的两个方法*/
#import "Person.h"
@implementation Person
// 解档
-(instancetype)initWithCoder:(NSCoder *)aDecoder
{
if (self = [super init]) {
self.name = [aDecoder decodeObjectForKey:@"name"];
self.age = [aDecoder decodeIntegerForKey:@"age"];
self.isMan = [aDecoder decodeBoolForKey:@"isMan"];
}
return self;
}
// 归档
-(void)encodeWithCoder:(NSCoder *)aCoder
{
[aCoder encodeObject:self.name forKey:@"name"];
[aCoder encodeInteger:self.age forKey:@"age"];
[aCoder encodeBool:self.isMan forKey:@"isMan"];
}
注意:
如果需要归档的类是某个自定义类的子类时,就需要在归档和解档之前先调用父类的[super initWithCoder:aDecoder]和[super encodeWithCoder:aCoder]方法;
/*使用*/
归档对象是调用NSKeyedArchiver的类方法archiveRootObject: toFile:方法,解档对象是调用NSKeyedUnarchiver的类方法unarchiveObjectWithFile:方法。
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
// 创建文件保存路径
NSString *personInfoPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).firstObject stringByAppendingPathComponent:@"personInfo.data"];
// 实例化一个person
Person *person = [Person initPersonWithName:@"宣晓" age:27 isMan:NO];
// 归档
[self archiverPerson:person filePath:personInfoPath];
// 解档
[self unarchiverPersonWithFilePath:personInfoPath];
}
#pragma mark - 归档对象
-(void)archiverPerson:(Person *)person filePath:(NSString *)filePath
{
// 归档
[NSKeyedArchiver archiveRootObject:person toFile:filePath];
}
#pragma mark - 解档对象
-(void)unarchiverPersonWithFilePath:(NSString *)filePath
{
// 解档
Person *person2 = [NSKeyedUnarchiver unarchiveObjectWithFile:filePath];
if (person2) {
NSLog(@"name:%@,age:%ld,isMan:%d",person2.name,(long)person2.age,person2.isMan);
}
}
注意:
使用归档和解档的类,必须遵循并实现NSCoding协议;
所保存文件的扩展名可以任意指定;
若有继承,必须先调用父类的归档和解档方法。