使用 NSPropertyListSerialization 持久化字典与数组

本文涉及的产品
密钥管理服务KMS,1000个密钥,100个凭据,1个月
简介:

NSPropertyListSerialization

The NSPropertyListSerialization class provides methods that convert property list objects to and from several serialized formats. Property list objects include NSData, NSString, NSArray, NSDictionary, NSDate, and NSNumber objects. These objects are toll-free bridged with their respective Core Foundation types (CFData, CFString, and so on). For more about toll-free bridging, see “Interchangeable Data Types”.

NSPropertyListSerialization这个类提供了一些方法,专门用来转换不同组织形式的list对象.list对象包括了NSData, NSString, NSArray, NSDictionary, NSDate, 以及 NSNumber.这些对象与衍生出他们的祖先对象有着一些联系,对于toll-free bridging,请看“Interchangeable Data Types”.

 

大家都应该使用过NSCoding协议来持久化一个对象,但你试着用NSCoding协议来持久化一个字典或者数组试一下.

问:为什么要持久化一个字典或者数组呢?

持久化一个字典或者数组,可以使用方法writeToFile:atomically:来写文件,但是有一个问题,这个只能写成plist文件,别人是可以通过明文来观看的,我可不喜欢别人能看清楚我存储了一些什么数据,你也是一样对吧.为什么要持久化一个字典或者数组呢?在需要存储的数据比较少时,此时又不需要数据库那种麻烦的查表方式来更改数据,更不需要CoreData这种大炮来打蚊子,所以,将数据存储于一个字典或者数组中便于维护以及轻量级,但是存储后成了plist文件,别人可以看到怎么办?那就把它存储成加密的NSData吧.

注:使用NSCoding协议是可以实现存储字典的,但是,那代码量你可以上网搜索一下解决方案,让人望而却步,本人在这里推荐使用下面的一个类NSPropertyListSerialization来处理字典或者数组的本地加密持久化,省掉你无数的代码.

使用非常简单,如下图所示:

注意:生成的NSData要记得进行加密后存储,提出出NSData时也需要进行解密操作后提取出来,推荐使用Des加密,看看有多简单.

 

附录:

经过本人一小时奋战,将NSPropertyListSerialization进行简易封装,附带源码以及使用教程,隐藏了NSPropertyListSerialization实现细节.

ListObjectOperation.h + ListObjectOperation.m

//
//  ListObjectOperation.m
//
//  Copyright (c) 2014年 Y.X. All rights reserved.
//

#import <Foundation/Foundation.h>

/**
 创建简单的文件路径
 
 @param filePath 简单的文件路径,如 @"/Documents/Y.X."
 
 @return 完整的沙盒文件路径
 */
NS_INLINE NSString * simpleFilePath(NSString *filePath)
{
    /*
     /Documents
     /Library/Caches
     /Library/Preferences
     /tmp
     */
    return [NSHomeDirectory() stringByAppendingString:filePath];
}

@interface ListObjectOperation : NSObject

/**
 将字典转换为NSData后进行存储
 
 @param dictionary 字典
 @param path 存储的路径
 
 @return none
 */
+ (void)storeDictionary:(NSDictionary *)dictionary toPath:(NSString *)path;

/**
 同步进行本地NSData以及字典同时操作
 
 @param path 字典NSData的路径
 @param flag 是否存储
 @param dictionary block中的字典,如果flag为YES,则修改的list值会被保存,flag为NO,则修改的值不会同步保存
 
 @return none
 */
+ (void)syncDictionaryWithDataPath:(NSString *)path
                              save:(BOOL)flag
                        dictionary:(void (^)(NSMutableDictionary *list))dictionary;

@end


//
//  ListObjectOperation.m
//
//  Copyright (c) 2014年 Y.X. All rights reserved.
//

#import "ListObjectOperation.h"

NS_INLINE NSData * dataFromFileUrlPath(NSString *path)
{
    return [NSData dataWithContentsOfURL:[NSURL fileURLWithPath:path]];
}

@implementation ListObjectOperation

+ (void)storeDictionary:(NSDictionary *)dictionary toPath:(NSString *)path
{
    NSData *data = [[self class] dataWithListObject:dictionary];
    [data writeToFile:path
           atomically:YES];
}

+ (void)syncDictionaryWithDataPath:(NSString *)path
                              save:(BOOL)flag
                        dictionary:(void (^)(NSMutableDictionary *list))dictionary
{
    // 获取字典
    NSDictionary *dic = [[self class] listObjectWithdata:dataFromFileUrlPath(path)];
    
    // 转化为可变字典
    NSMutableDictionary *changeDic = [NSMutableDictionary dictionaryWithDictionary:dic];
    
    if (flag == YES)
    {
        // 传入到字典中
        dictionary(changeDic);
     
        // 生成二进制文件
        NSData *data = [[self class] dataWithListObject:changeDic];
        
        // 进行存储(覆盖了之前的文件)
        [data writeToFile:path
               atomically:YES];
    }
    else
    {
        // 传入到字典中
        dictionary(changeDic);
    }
}

+ (NSData *)dataWithListObject:(id)listObject
{
    return [NSPropertyListSerialization dataWithPropertyList:listObject
                                                      format:NSPropertyListBinaryFormat_v1_0
                                                     options:0
                                                       error:nil];
}

+ (id)listObjectWithdata:(NSData *)data
{
    return [NSPropertyListSerialization propertyListWithData:data
                                                     options:0
                                                      format:NULL
                                                       error:nil];
}

@end

第一步:进行存储操作(请君先执行一遍程序,生成一个二进制文件 .YouXianMing 存储在沙盒的 /Document文件夹下)

第二步:同步修改字典以及存储二进制文件(请君再一次运行程序)

第三步:验证是否存储进去了

担心效率吗?不用担心,才2ms而已.实际上我测试了同时修改和存储,也就3ms,轻量级存储实在是太简单了^_^.

目录
相关文章
|
3月前
|
存储 JSON NoSQL
Redis第五弹-HASH结构相关指令和介绍,计数功能Hash-哈希(Redis本来就是键值对结构,哈希,就相当于键值对嵌套了一个键值对)的多种指令Hset key field value-
Redis第五弹-HASH结构相关指令和介绍,计数功能Hash-哈希(Redis本来就是键值对结构,哈希,就相当于键值对嵌套了一个键值对)的多种指令Hset key field value-
集合、哈希表、字典
集合、哈希表、字典
65 0
|
4月前
|
存储 缓存 算法
数据结构与算法面试题:实现一个 LRU 缓存,支持如下操作:获取值、更新值、删除键值对和插入键值对
数据结构与算法面试题:实现一个 LRU 缓存,支持如下操作:获取值、更新值、删除键值对和插入键值对
62 0
|
4月前
|
存储 JSON NoSQL
Redis中当存储数据为List集合时,如何控制集合内每个数据元素的生命周期
Redis中当存储数据为List集合时,如何控制集合内每个数据元素的生命周期
341 0
|
存储 缓存 NoSQL
键值对的集合:深入了解 Redis 的 Hash 数据类型
在现代的应用程序中,复杂数据结构的存储和快速访问对于高效的数据管理至关重要。Redis,作为一款高性能的内存数据库,提供了多种数据类型来满足不同的需求。在本文中,我们将着重介绍 Redis 的 Hash 数据类型,探讨其特性、用法以及在实际应用中的优势。
119 0
|
Python
一日一技:让你的字典读取不报错
一日一技:让你的字典读取不报错
74 0
一日一技:让你的字典读取不报错
|
存储 索引 Python
12.从入门到精通:Python字典,创建字典,访问字典的值,修改字典,删除字典,字典键的特性,字典内置函数和方法
12.从入门到精通:Python字典,创建字典,访问字典的值,修改字典,删除字典,字典键的特性,字典内置函数和方法
|
缓存 NoSQL Dubbo
因在缓存对象中增加字段,而导致Redis中取出缓存转化成Java对象时出现反序列化失败的问题...
因在缓存对象中增加字段,而导致Redis中取出缓存转化成Java对象时出现反序列化失败的问题...
270 0
|
开发者 Python
字典的遍历|学习笔记
快速学习字典的遍历
字典的遍历|学习笔记
对象转字典、字典排序
对象转字典、字典排序
98 0