iOS中常见的数据数据存储方式及特点

2016-03-23  本文已影响249人  forvert

iOS中常见的数据数据存储方式及特点

方式一: XML属性列表(plist)归档

属性列表是一种XML格式的文件,拓展名为plist

方式一: Preference(偏好设置)

本质还是通过“plist”来存储数据, 但是使用更简单(无需关注文件、文件夹路径和名称)

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

方式三: NSKeyedArchiver归档(NSCoding)

把任何对象, 直接保存为文件的方式。
如果对象是NSString、NSDictionary、NSArray、NSData、NSNumber等类型,可以直接用NSKeyedArchiver进行归档和恢复

//  一个数组
NSArray *array = [NSArray arrayWithObjects:@”a”,@”b”,nil];
// 归档(编码)NSArray对象
[NSKeyedArchiver archiveRootObject:array toFile:path];
// 恢复(解码)NSArray对象
NSArray *array = [NSKeyedUnarchiver unarchiveObjectWithFile:path];

方式四: SQLite3(数据库)

使用场景:当非常大量的数据存储时使用

注意: 在iOS中使用SQLite3,首先要添加库文件libsqlite3.dylib和导入主头文件

方式五: Core Data

本质就是对SQLite的封装

使用此功能,要添加CoreData.framework和导入主头文件<CoreData/CoreData.h>

补充一:SQLite

什么是数据库?

存储数据的仓库,说白了就是一个文件;

数据库存储数据的步骤?

主键的设置原则

SQL语句(结构化查询语言)

特点
关键字
SQL语句种类

补充二:数据持久化

plist文件(又称属性列表文件)

NSString *path = [[NSBundle mainBundle]pathForResource:@"" ofType:@""];
// 每一个NSBundle对象对应一个资源包
[NSBundle mainBundle] 是代表主资源包;

你会如何存储用户的一些敏感信息,如登录的 token


#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];
}
上一篇 下一篇

猜你喜欢

热点阅读