iOS应用数据存储

2016-12-03  本文已影响0人  SevEnOye

在iOS开发中常用的方式有如下几种:

  1. XML属性列表(plist)归档

实现数据存储之前,首先要先了解什么是应用沙盒。

什么是应用沙盒?

应用沙盒结构的怎么样的?

应用沙盒目录的常见获取方式

 //利用沙盒根目录拼接”Documents”字符串
NSString *home = NSHomeDirectory();
NSString *documents = [home stringByAppendingPathComponent:@"Documents"];

方法二

// 利用NSSearchPathForDirectoriesInDomains函数
// NSUserDomainMask代表从用户文件夹下找
// YES 代表展开路径中的波浪字符“~”
NSArray *array =  NSSearchPathForDirectoriesInDomains(NSDocumentDirectory,
NSUserDomainMask,NO);
// 在iOS中,只有一个目录跟传入的参数匹配,所以这个集合里面只有一个元素
NSString *documents = [array objectAtIndex:0];
  1. 利用沙盒根目录拼接”Caches”字符串

什么是属性列表?

plist存储

plist存储本质就是帮你生成一个plist文件,可以将数组或者字典数据保存到plist文件中。
Ps: plist的数据不是数组就是字典,并且不能存储自定义对象

Demo
- (void)writeToPlist{
    NSArray *arr = @[@"123",@1];
    
    // 获取Caches文件夹路径
    // directory:搜索文件夹
    // domainMask:在哪个范围内搜索 NSUserDomainMask:在用户中查找
    // expandTilde: YES :在路径展开 NO:不展开路径 ~:代替沙盒路径
    NSString *cachesPath = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)[0];
    
    // 拼接文件名
    NSString *filePath = [cachesPath stringByAppendingPathComponent:@"arr.plist"];    
    [arr writeToFile: filePath atomically:YES];
}

- (void)redFromPlist {
    NSString *cachesPath = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)[0];
    NSString *filePath = [cachesPath stringByAppendingPathComponent:@"arr.plist"];
    
    // 根据存储的形式选择相同的形式读取
    NSArray *arr = [NSArray arrayWithContentsOfFile:filePath];  
    ZGLog(@"%@", arr);
}

偏好设置

偏好设置有什么作用?
偏好设置的好处:
  1. 快速进行键值对的存储
Demo
- (void)save {
    // 获取NSUserDefaults单例对象
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    [defaults setObject:@"ZoengGit" forKey:@"name"];   
    [defaults setObject:@"18" forKey:@"age"];

}

- (void)read {  
    NSString *name = [[NSUserDefaults standardUserDefaults] objectForKey:@"name"];
    NSString *age = [[NSUserDefaults standardUserDefaults] objectForKey:@"age"];
    NSLog(@"%@, %@", name , age);
}

注意:UserDefaults设置数据时,不是立即写入,而是根据时间戳定时地把缓存中的数据写入本地磁盘。所以调用了set方法之后数据有可能还没有写入磁盘应用程序就终止了。出现以上问题,可以通过调用synchornize方法强制写入
[defaults synchornize];

NSKeyedArchiver

NSCoding协议有2个方法:
encodeWithCoder:
每次归档对象时,都会调用这个方法。一般在这个方法里面指定如何归档对象中的每个实例变量,可以使用encodeObject:forKey:方法归档实例变量

initWithCoder:
每次从文件中恢复(解码)对象时,都会调用这个方法。一般在这个方法里面指定如何解码文件中的数据为对象的实例变量,可以使用decodeObject:forKey方法解码实例变量
// 归档一个NSArray对象到Documents/array.archive
NSArray *array = [NSArrayarrayWithObjects:@”a”,@”b”,nil];
[NSKeyedArchiverarchiveRootObject:arraytoFile:path];

// 恢复(解码)NSArray对象
NSArray *array = [NSKeyedUnarchiverunarchiveObjectWithFile:path];
Demo

Person.h文件

@interface
Person : NSObject<NSCoding>
@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@property (nonatomic, assign) float height;
@end

Person.m文件

@implementation Person

// 什么作用:告诉系统模型中的哪些属性需要归档
// 什么时候调用:把一个自定义对象,归档的时候调用
- (void)encodeWithCoder:(NSCoder*)encoder {
    [encoder encodeObject:self.nameforKey:@"name"];
    [encoder encodeInt:self.ageforKey:@"age"];
    [encoder encodeFloat:self.heightforKey:@"height"];
}

// 什么时候调用:解析一个文件的时候调用
// 作用:告诉系统模型中的哪些属性需要解档
- (id)initWithCoder:(NSCoder*)decoder {
    self.name = [decoder decodeObjectForKey:@"name"];
    self.age = [decoder decodeIntForKey:@"age"];
    self.height = [decoder decodeFloatForKey:@"height"];
    return self;
}

- (void)dealloc{
    [super dealloc];
    [_name release];
}
@end

归档Person对象(编码和解码)

// 归档(编码)
Person *person = [[[Person alloc] init] autorelease];
person.name = @"ZoengGit";
person.age = 18;
person.height = 1.83f;

[NSKeyedArchiver archiveRootObject:person toFile:path];

//恢复(解码)
Person *person = [NSKeyedUnarchiver unarchiveObjectWithFile:path];

归档对象的注意点

1. 如果父类也遵守了NSCoding协议,请注意:应该在encodeWithCoder:方法中加上如下代码, 确保继承的实例变量也能被编码,即也能被归档
self = [super encodeWithCode:encode];

2. 应该在initWithCoder:方法中加上如下代码确保继承的实例变量也能被解码,即也能被恢复
self = [super initWithCoder:decoder];

上一篇 下一篇

猜你喜欢

热点阅读