App开发流程之数据持久化和编译静态链接库

Posted 吾将上下而求索

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了App开发流程之数据持久化和编译静态链接库相关的知识,希望对你有一定的参考价值。

先记录数据持久化。

ios客户端提供的常用数据持久化方案:NSUserDefaults代表的用户设置,NSKeydArchiver代表的归档,plist文件存储,SQLite数据库(包括上层使用的Core Data,FMDB)。

每种方案都有各自的应用场景和范围,不能一概而论。不过可以大致以数据储存量和复杂度来区别。

除了以上提到的方案,再记录一种方案:LevelDB代表的键值对数据库。

 

NSUserDefaults常用方法:

1.可以使用标准用户设置[NSUserDefaults standardUserDefaults],也可以通过init相关方法初始化新的用户设置

2.像使用字典一样获取、设置、移除键值对

3.synchronize方法已经不建议使用

 

Plist文件存储:

1.代码读取应用内已经存在的plist文件,得到一个字典

    NSString *filePath = [[NSBundle mainBundle] pathForResource:@"test" ofType:@"plist"];

    NSMutableDictionary *dic = [[NSMutableDictionary alloc] initWithContentsOfFile:filePath];

2.修改数据后,保存或者创建plist文件

    [dic writeToFile:filePath atomically:YES];

 

NSKeydArchiver和NSKeyedUnarchiver:

1.归档有一个类方法:+ (BOOL)archiveRootObject:(id)rootObject toFile:(NSString *)path;

   解档有一个类方法:+ (nullable id)unarchiveObjectWithFile:(NSString *)path;

 可以直接对某一个对象进行归档和解档。

2.但如果需要对多个键值对进行操作,建议使用如下方法:

+ (void)archiveDataWithDictionary:(NSDictionary *)dic filename:(NSString*)filename archiveSuccessBlock:(archiveSuccessBlock)archiveSuccessBlock
{
    NSString *fullPath = [self getAppArchivedFileFullPathWithName:filename];
    
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        NSMutableData *data = [NSMutableData data];
        NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
        
        NSArray *keyArray = [NSArray arrayWithArray:[dic allKeys]];
        [archiver encodeObject:keyArray forKey:filename];
        
        for (NSString *key in keyArray) {
            NSObject *object = [dic objectForKey:key];
            [archiver encodeObject:object forKey:key];
        }
        
        [archiver finishEncoding];
        [data writeToFile:fullPath atomically:YES];
        
        if (archiveSuccessBlock) {
            archiveSuccessBlock();
        }
    });
}

+ (NSDictionary *)unarchiveDataWithFilename:(NSString *)filename
{
    NSMutableDictionary *dic = [NSMutableDictionary dictionary];
    
    NSData *data = [[NSData alloc] initWithContentsOfFile:[self getAppArchivedFileFullPathWithName:filename]];
    NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
    
    NSArray *keyArray = [NSArray arrayWithArray:[unarchiver decodeObjectForKey:filename]];
    
    for (NSString *key in keyArray) {
        NSObject *object = [unarchiver decodeObjectForKey:key];
        [dic setObject:object forKey:key];
    }
    
    [unarchiver finishDecoding];

    return dic;
}

归档的initForWritingWithMutableData和finishEncoding,解档的initForReadingWithData和finishDecoding需要成对出现。

 

SQLite数据库:

只要使用过SQL Server和mysql之类的关系型数据库,就可以轻松使用,只是底层的sql语言不太人性化,所以普遍采用了上层的Core Data或者FMDB类库。

 

文件操作:

首先关注方法:FOUNDATION_EXPORT NSArray<NSString *> *NSSearchPathForDirectoriesInDomains(NSSearchPathDirectory directory, NSSearchPathDomainMask domainMask, BOOL expandTilde);第一个枚举参数表示文件目录,第二个表示范围域,第三个参数表示是否补充完整的相对路径

1.得到当前用户的doc文件根目录:

  NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);

    NSString *rootPath = paths[0];

2.补充子文件路径

    NSString *fullPath = [rootPath stringByAppendingPathComponent:filename];

3.文件操作,主要使用[NSFileManager defaultManager]单例对象 

+ (void)createArchivedRootFile
{
    NSString *rootPath = [self getAppArchivedFilesRootPath];
    
    if (![[NSFileManager defaultManager] fileExistsAtPath:rootPath]) {
        [[NSFileManager defaultManager] createDirectoryAtPath:rootPath withIntermediateDirectories:YES attributes:nil error:nil];
    }
}

+ (void)clearArchivedFileWithName:(NSString *)filename
{
    NSString *rootPath = [self getAppArchivedFilesRootPath];
    NSString *fullPath = [rootPath stringByAppendingPathComponent:filename];
    
    if ([[NSFileManager defaultManager] fileExistsAtPath:fullPath]) {
        [[NSFileManager defaultManager] removeItemAtPath:fullPath error:nil];
    }
}

 

除了以上记录方案,再记录一下使用键值对数据库LevelDB的经历。

当数据量并不是很大,但是又需要数据库存储和操作时候,键值对数据库是首选。源自Google的LevelDB是其中的明星。

之前遇到客户端存储省市区地址数据的需求,并当用户选择地址时候读取相关数据。如果每次都完整读取省市区数据,占用内存既大又没有必要,因为用户很可能只会选择一个省下的一个市的一个区。

如果将省市区数据拆分为若干键值对,并且建立某种链式关系,就可以将数据以键值对分散存储于某个地方,并且快速读取需要数据。

1.第一层级只有一个键值对,key为固定值,value为全部省的名称数组

2.第二层级键值对数量为省数量,key为省名称,value为市名称数组

3.第三层级键值对数量为市数量,key为“省名称.市名称”,value为区名称数组

4.。。。。。

如上,数据全部以键值对分散存储于LevelDB中,只要知道key规则和名称,就可以快速取到对应数据,而优秀的IO保证了性能表现。

这是之前记录的一篇关于LevelDB的文章,可以先参考一下:http://www.cnblogs.com/A-Long-Way-Chris/p/4864573.html

 

编译静态链接库

正好以LevelDB为案例。先前往下载C++源代码: https://github.com/google/leveldb

使用Xcode创建静态链接库

1.新建项目,选择类型

 

2.设置项目Build Phases,点击区域左上角加号,选择添加Headers Phase

 

3.点击加号,添加需要公开暴露的头文件,然后从Project栏拖拽到Public栏

 

 

 4.切换真机和模拟器,分别编译成功后,右键Products目录下的libleveldb.a,在Finder中查看

 

5.在终端程序中cd到该目录,输入如下指令,即可导出同时支持真机和模拟器运行的静态链接库

lipo -create Debug-iphoneos/libleveldb.a Debug-iphonesimulator/libleveldb.a -output libleveldb.a 

 

使用命令行,通过Makefile编译LevelDB的静态链接库

1.解压下载包后,使用Visual Studio Code之类的文本编辑器打开目录下Makefile

2.修改Makefile中的CXXFLAGS,添加指令 -fembed-bitcode,保存

 

3.在终端中,cd到LevelDB目录,输入指令:CXXFLAGS=-miphoneos-version-min=7.0 make PLATFORM=IOS

表示生成iOS版本的静态链接库,支持最低版本为7.0(该设置保证在模拟器上可以正常运行)。

如果提示permission denied,则在上述指令前加上sudo,最终为:sudo CXXFLAGS=-miphoneos-version-min=7.0 make PLATFORM=IOS,然后输入密码回车即可。

 

 4.说明一下,如果跳过步骤1和2,最后生成的LevelDB静态链接库不支持bitcode,可以看到体积相差还是比较大的,按需编译

 

将.a文件和include目录下的头文件加入项目即可正常使用。

1.如果遇到提示某头文件找不到,请检查项目配置中,Header Search Paths是否有配置缺失

2.如果使用不支持bitcode的版本,需要在build setting中将enable bitcode设置为NO。该设置对其他类库要求一样

 

为了便于使用OC编程,还引入了另一个类库,对LevelDB的代码进行了OC封装,地址:https://github.com/matehat/Objective-LevelDB

 

我在base项目中,增加了LevelDBHelper工具类,进一步对调用代码进行了封装,操作更简单安全。

 

base项目已更新:git@github.com:ALongWay/base.git

以上是关于App开发流程之数据持久化和编译静态链接库的主要内容,如果未能解决你的问题,请参考以下文章

gcc/g++实战之动态链接库与静态链接库编写

C开发编译与调试

C开发编译与调试

编译链接实战(16)静态库vs动态库

计算机系统篇之链接:静态链接(上)

计算机系统篇之链接:静态链接(上)