iOS中常见的数据数据存储方式及特点
方式一: XML属性列表(plist)归档
属性列表是一种XML格式的文件,拓展名为plist
- 如果对象是NSString、NSDictionary、NSArray、NSData、NSNumber等类型,就可以使用
writeToFile:atomically:
方法直接将对象写到属性列表文件.
方式一: Preference(偏好设置)
本质还是通过“plist”来存储数据, 但是使用更简单(无需关注文件、文件夹路径和名称)
- 很多iOS应用都支持偏好设置,比如保存用户名、密码、字体大小等设置,iOS提供了一套标准的解决方案来为应用加入偏好设置功能,如下:
-
每个应用都有个NSUserDefaults实例,通过它来存取偏好设置
比如,保存用户名、字体大小、是否自动登录// 获取偏好设置单例对象 NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; // 以username为key,以ljforever为value存储 [defaults setObject:@"ljforever" forKey:@"username"]; // 以text_size为key,以18.0f为value存储 [defaults setFloat:18.0f forKey:@"text_size"]; // 以auto_login为key,以YES为value存储 [defaults setBool:YES forKey:@"auto_login"];
-
读取上次保存的设置
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; NSString *username = [defaults stringForKey:@"username"]; float textSize = [defaults floatForKey:@"text_size"]; BOOL autoLogin = [defaults boolForKey:@"auto_login"];
注意:UserDefaults设置数据时,不是立即写入,而是根据时间戳定时地把缓存中的数据写入本地磁盘。所以调用了set方法之后数据有可能还没有写入磁盘应用程序就终止了。出现以上问题,可以通过调用synchornize方法强制写入
[defaults synchornize];
方式三: NSKeyedArchiver归档(NSCoding)
把任何对象, 直接保存为文件的方式。
如果对象是NSString、NSDictionary、NSArray、NSData、NSNumber等类型,可以直接用NSKeyedArchiver进行归档和恢复
注意:不是所有的对象都可以直接用这种方法进行归档,只有遵守了NSCoding协议的对象才可以
-
NSCoding协议有2个方法:
encodeWithCoder:
- 每次归档对象时,都会调用这个方法。一般在这个方法里面指定如何归档对象中的每个实例变量,可以使用
encodeObject:forKey:
方法归档实例变量. -
initWithCoder:
每次从文件中恢复(解码)对象时,都会调用这个方法。一般在这个方法里面指定如何解码文件中的数据为对象的实例变量,可以使用decodeObject:forKey
方法解码实例变量
例如:归档一个NSArray对象到Documents/array.archive
// 一个数组
NSArray *array = [NSArray arrayWithObjects:@”a”,@”b”,nil];
// 归档(编码)NSArray对象
[NSKeyedArchiver archiveRootObject:array toFile:path];
// 恢复(解码)NSArray对象
NSArray *array = [NSKeyedUnarchiver unarchiveObjectWithFile:path];
方式四: SQLite3(数据库)
使用场景:当非常大量的数据存储时使用
SQLite3是一款开源的嵌入式关系型数据库
-
优点:
- 可移植性好
- 易使用、内存开销小
-
SQLite3是无类型的,意味着你可以保存任何类型的数据到任意表的任意字段中。
比如:下列的创表语句是合法的:
create table t_person(name, age);
为了保证可读性,建议还是把字段类型加上:
create table t_person(name text, age integer);
SQLite3常用的
5种数据类型
:text
、integer
、float
、boolean
、blob
注意: 在iOS中使用SQLite3,首先要添加库文件libsqlite3.dylib和导入主头文件
方式五: Core Data
本质就是对SQLite的封装
- Core Data框架提供了
对象-关系映射(ORM)
的功能, - 即能够将OC对象转化成数据,保存在SQLite3数据库文件中,也能够将保存在数据库中的数据还原成OC对象。在此数据操作期间,不需要编写任何SQL语句。
使用此功能,要添加CoreData.framework和导入主头文件<CoreData/CoreData.h>
补充一:SQLite
什么是数据库?
存储数据的仓库,说白了就是一个文件;
数据库存储数据的步骤?
- 新建一个数据库文件
- 新建一张表
- 添加多个字段(column,列,属性)
- 添加多行的记录(row,每行存储多个字段对应的值)
主键的设置原则
- 主键应当对用户没有任何意义
- 永远不要跟新主键
- 主键不应包含动态变化的数据
- 主键应有计算机自动生成
SQL语句(结构化查询语言)
特点
- 不区分大小写
- 每句语句都是以';'结尾
关键字
- select、insert、update、delete、from、create、where、desc、order、by、group、table、alter、view、index等等
SQL语句种类
-
数据定义语句(DDL)
- 包括create和drop操作(创表,删除表格)
- 在数据库中创建新表或删除表(create table或 drop table)
-
数据操作语句(DML:Data Manipulation Language)
- 包括insert、update、delete等操作
- 上面的3种操作分别用于添加、修改、删除表中的数据
-
数据查询语句(DQL:Data Query Language)
- 可以用于查询获得表中的数据
- 关键字select是DQL(也是所有SQL)用得最多的操作
- 其他DQL常用的关键字有where,order by,group by和having
补充二:数据持久化
plist文件(又称属性列表文件)
- 加载plist文件
- [根节点类型 类型WithContentOfFile:全路径];
- 在iOS中看见file就是全路径;
- 利用NSBundle获取全路径;
NSString *path = [[NSBundle mainBundle]pathForResource:@"" ofType:@""];
// 每一个NSBundle对象对应一个资源包
[NSBundle mainBundle] 是代表主资源包;
- plist文件使用注意点
你会如何存储用户的一些敏感信息,如登录的 token
使用keychain来存储,也就是钥匙串,使用keychain需要导入Security框架
自定义一个keychain的类
#import <Security/Security.h>
@implementation YCKKeyChain
+ (NSMutableDictionary *)getKeychainQuery:(NSString *)service {
return [NSMutableDictionary dictionaryWithObjectsAndKeys:
(__bridge_transfer id)kSecClassGenericPassword,(__bridge_transfer id)kSecClass,
service, (__bridge_transfer id)kSecAttrService,
service, (__bridge_transfer id)kSecAttrAccount,
(__bridge_transfer id)kSecAttrAccessibleAfterFirstUnlock,(__bridge_transfer id)kSecAttrAccessible,
nil];
}
+ (void)save:(NSString *)service data:(id)data {
// 获得搜索字典
NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];
// 添加新的删除旧的
SecItemDelete((__bridge_retained CFDictionaryRef)keychainQuery);
// 添加新的对象到字符串
[keychainQuery setObject:[NSKeyedArchiver archivedDataWithRootObject:data] forKey:(__bridge_transfer id)kSecValueData];
// 查询钥匙串
SecItemAdd((__bridge_retained CFDictionaryRef)keychainQuery, NULL);
}
+ (id)load:(NSString *)service {
id ret = nil;
NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];
// 配置搜索设置
[keychainQuery setObject:(id)kCFBooleanTrue forKey:(__bridge_transfer id)kSecReturnData];
[keychainQuery setObject:(__bridge_transfer id)kSecMatchLimitOne forKey:(__bridge_transfer id)kSecMatchLimit];
CFDataRef keyData = NULL;
if (SecItemCopyMatching((__bridge_retained CFDictionaryRef)keychainQuery, (CFTypeRef *)&keyData) == noErr) {
@try {
ret = [NSKeyedUnarchiver unarchiveObjectWithData:(__bridge_transfer NSData *)keyData];
} @catch (NSException *e) {
NSLog(@"Unarchive of %@ failed: %@", service, e);
} @finally {
}
}
return ret;
}
+ (void)delete:(NSString *)service {
NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];
SecItemDelete((__bridge_retained CFDictionaryRef)keychainQuery);
}
- 在别的类实现存储,加载,删除敏感信息方法
// 用来标识这个钥匙串
static NSString * const KEY_IN_KEYCHAIN = @"com.yck.app.allinfo";
// 用来标识密码
static NSString * const KEY_PASSWORD = @"com.yck.app.password";
+ (void)savePassWord:(NSString *)password
{
NSMutableDictionary *passwordDict = [NSMutableDictionary dictionary];
[passwordDict setObject:password forKey:KEY_PASSWORD];
[YCKKeyChain save:KEY_IN_KEYCHAIN data:passwordDict];
}
+ (id)readPassWord
{
NSMutableDictionary *passwordDict = (NSMutableDictionary *)[YCKKeyChain load:KEY_IN_KEYCHAIN];
return [passwordDict objectForKey:KEY_PASSWORD];
}
+ (void)deletePassWord
{
[YCKKeyChain delete:KEY_IN_KEYCHAIN];
}