前面已经讲解了SQLite,FMDB以及CoreData的基本操作和代码讲解(CoreData也在不断学习中,上篇博客也会不断更新中)。本篇我们将讲述在实际开发中,所使用的iOS数据持久化的方式以及怎么会使用到这些方式,都会以本人实际开发的场景为例,大约需要花10-15分钟,欢迎大家指正。
一、前言
和大家说一个真实故事,前年我去美图面试(当时的技术仅仅是UI和接口的实现,并不注重很多底层实现和很多概念的原理,换句话说,就是真正的码农),过了技术第一轮和第二轮(前两年的也就是问问技术点的实现),到了第三轮的时候,应该是项目经理面试我,其中一个题目问住了我,因为iOS我是自学的,也算是不断摸索的。
“请你说一下iOS 一般会默认自己程序的目录,请说一下这个目录机构,以及组成部分,里面包含什么?”
当时的情景就是,我是谁,我在哪,大家也可以想一下!!!
其实那个项目经理问的就是我们经常听到的沙盒文件,其中iOS在程序默认下只会访问自己的程序目录-也就是沙盒文件。
1. 寻找沙盒文件位置
大家可能不知道怎么查看正在真机(或者模拟器)运行下的沙盒文件,可以按照下面的步骤查看
1.1 第一步,打开项目,选择windows(我是以真机为例,模拟器亦是)
1.2 第二步,进入界面,选择自己的app,点击绿色按钮,
1.3 第三步,选择第二个Download Container,导出到指定的文件夹(我是导出到桌面)
1.4 最后,我们发现桌面有个后缀名为.xcappdata的文件夹,我是导入到桌面。
2.沙盒文件目录结构
2.1 在1.4中后缀.xcappdata,右击选择->显示包内容,打开看到目录如下图,也可以继续点击查看更详细内容。
2.2目录讲解
:->AppData:也就是应用程序包,存的是程序的源文件,其中里面有可执行的文件以及资源文件。通过以下可以获得路径
NSString *path = [[NSBundle mainBundle] bundlePath];
NSLog(@"%@", path);
AppData里面包括了以下内容:
(1)Documents
也是我们经常使用到的目录,我们经常看到iTunes同步,同步的就是此文件中的内容,Documents适合存比较重要的数据。我们可以通过以下代码拿到Documents
NSString *path = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).firstObject;
NSLog(@"%@", path);
(2)Library
Library比较适合存储比较大的数据,并且iTunes不会同步此文件,也不会备份。可以通过以下拿到路径
NSString *path = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES).firstObject;
NSLog(@"%@", path);
(2.1)获取Caches目录路径
NSString *cachesDir = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
NSLog(@"Caches目录路径是%@",docDir);
(2.2)Preferences
主要包括了程序中的偏好设置文件,下面会讲述这个Preferences
(3)SystemData
目前暂不讲解这个目录(里面目录为空)
(4)tmp
用于存储临时文件,主要保存程序再次启动中不需要的信息数据,并且该路径中的文件并不会被itunes同步和备份。通过以下获得
NSString *tmpDir = NSTemporaryDirectory();
上面就是前言部分,大家这篇博客,我们主要讲述iOS数据持久化操作,好,我们正式以项目的形式讲述iOS的数据持久化操作。
二、iOS数据持久化操作
对于iOS的数据持久化操作,在网上也是有很多讲解,本篇主要讲述本人项目中使用到的方式进行讲解。iOS持久化操作方式主要有以下五种方式:
1. plist文件方式,也就是属性列表
2. preference 也就是偏好设置
3. NSKeyedArchiver 也就是归档
4. SQLite
5. CoreData
下面我们就一一讲解五种方式,以项目为例。
2.1 plist (属性列表)
plist是将比较特定的类(不经常改变的类),通过XML的方式存储在目录中。
自己项目有个分享的功能,因为分享渠道是固定的以及界面内容不会改变,所以自己当初采取plist文件的方式进行存储。效果如下:
具体分享平台自己使用plist文件进行存储,下面是自己plist文件
点开plist文件,如下图
右击该plist文件->选择Open As ->选择Source Code,如下
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>data</key>
<array>
<dict>
<key>titleLabel</key>
<string>微信好友</string>
<key>imageView</key>
<string>cart_wx</string>
</dict>
<dict>
<key>titleLabel</key>
<string>朋友圈</string>
<key>imageView</key>
<string>cart_friend</string>
</dict>
<dict>
<key>titleLabel</key>
<string>QQ好友</string>
<key>imageView</key>
<string>cart_qq</string>
</dict>
<dict>
<key>titleLabel</key>
<string>QQ空间</string>
<key>imageView</key>
<string>cart_que</string>
</dict>
</array>
</dict>
</plist>
下面讲述怎么使用分享plist文件。
通过上图和XML文件可以看出最外面是字典,我在项目是使用懒加载方式加载这个字典
代码如下:
- (NSDictionary *)configDatas{
if (!_configDatas) {
NSString *path = [[NSBundle mainBundle]pathForResource:@"IOACartShareDatas" ofType:@".plist"];
NSDictionary *dic = [NSDictionary dictionaryWithContentsOfFile:path];
_configDatas = dic;
}
return _configDatas;
}
在分享的view内进行传值
然后进入shareView中查看具体赋值,布局采用了UICollectionView布局
补充一下:
自己上面是一个一个在plist里面创建的元素,也可以通过writeToFile存储以及arrayWithContentsOfFile取文件。
(1)writeToFile存储方式
NSArray *array = @[@"2", @"1", @"3"];
[array writeToFile:fileName atomically:YES];
(2)arrayWithContentsOfFile读取方式
NSArray *result = [NSArray arrayWithContentsOfFile:fileName];
NSLog(@"%@", result);
2.2 Preference偏好设置
Preference偏好设置一般用存储应用程序的配置信息文件,最好不要在Preference中存储其他数据,Preference通过操作NSUserDefaults完成操作。
2.2.1 获得NSUserDefaults文件
//获得NSUserDefaults文件
NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
2.2.2 开始向文件中写入内容
[userDefaults setObject:@"哈哈哈" forKey:@"a"];
[userDefaults setBool:YES forKey:@"success"];
[userDefaults setInteger: forKey:@"age"];
2.2.3 确定是否同步。确定是否调用synchronize方法,想要写入文件时,就应该调用[userDefaults synchronize]方法
如果同步
[userDefaults synchronize];
反之,不写。
2.2.4 在合适的地方调取文件
NSString *name = [userDefaults objectForKey:@"a"];
BOOL sex = [userDefaults boolForKey:@"success"];
NSInteger age = [userDefaults integerForKey:@"age"];
NSLog(@"%@, %d, %ld", name, sex, age);
Preference会把所有的数据存储到一个文件中,也就是在上面preference文件下的plist文件。
2.3 NSKeyedArchiver 归档
在使用归档NSKeyedArchiver一定要遵守NSCoding协议,也就是遵守了NSCoding协议的对象,我们都可以通过归档NSKeyedArchiver进行序列化。
通过查看NSCoding协议文件,发现如下
NSCoding协议文件声明了两个文件,上述图中,也是必须要实现的。encodeWithCoder用于将对象编码到归档中,而initWithCoder用于通过解档来获取一个新对象。
需要注意的是:
假如要归档的类恰好是一个自定义类的子类时候,就应该要在归档和解档之前要首先实现父类的方法,也就是必须要实现以下两个方法:[super encodeWithCoder:aCoder]和[super initWithCoder:aDecoder]
拓展->
NSCopying协议文件如下(有一次面试官问我,NSCopying与NSMutableCopying),大家当增长一个知识点吧!
下面以自己以前的项目为准,讲述归档的使用(因为以前项目较小,又是独立开发,采用NSKeyedArchiver存储信息)。
2.3.1 使用NSKeyedArchiver归档存储用户登录部分信息,注销登录清除。定义了一个类AppUserDefaults
点开AppUserDefaults.h文件定义登录获取的属性(仅仅展示部分属性,实现都一样)
通过以下方法对登录信息的赋值
我们看一下归档和解档,以及上面方法的实现(注解上面type属性是代表是不同短登录:例如医生端和患者端)
2.3.2 encodeWithCoder归档(显示部分属性,否则界面太大)
2.3.2 initWithCoder解档
2.3.2 登录通过setSessionWithLoginName设置属性
+ (void)setSessionWithLoginName:(NSString *)loginName
version:(NSString *)version
loginTime:(long long)loginTime
id_:(long)id_
userType:(int)userType
name:(NSString *)name
phone:(NSString *)phone
pic:(NSString *)pic
companyName:(NSString *)companyName
address:(NSString *)address
type:(NSInteger)type
cityName:(NSString *)cityName
cityId:(long)cityId
latitude:(NSString *)latitude
longitude:(NSString *)longitude {
//判断版本号,通过session
AppUserDefaults *session = [AppUserDefaults getSession];
if (session && ![session.version isEqualToString:@""]) {
if (type != -)
session.type = type;
if (loginTime != -)
session.loginTime = loginTime;
if ([loginName isNotBlank])
session.loginName = loginName;
if ([version isNotBlank])
session.version = version;
if (id_ != -)
session.id_ = id_;
}
NSData *data = [NSKeyedArchiver archivedDataWithRootObject:session];
if (data) {
NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
[user setObject:data forKey:@"session"];
[user synchronize];
}
} + (AppUserDefaults *)getSession {
AppUserDefaults *session = nil;
NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
NSData *data = [user objectForKey:@"session"];
if (data) {
session = [NSKeyedUnarchiver unarchiveObjectWithData:data];
if (session && [session.version isEqualToString:@""]) {
session = nil;
}
}
return session;
}
2.3.4 上面就是AppUserDefaults文件内容,下面讲述在登录时候进行赋值。
首先要懒加载AppUserDefaults
- (AppUserDefaults *)userDefaults{
if (_userDefaults == nil) {
_userDefaults = [AppUserDefaults getSession];
}
return _userDefaults;
}
登录成功时赋值
2.3.5 如果用户注销之后或者退出应用应该给重新清空,如下:
总结:
使用NSKeyedArchiver的工厂方法[NSKeyedArchiver archiveRootObject:obj toFile:path];方法。demo如下:
NSString *file = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).firstObject stringByAppendingPathComponent:@"student.data"];
Person *person = [[Person alloc] init];
person.avatar = self.avatarView.image;
person.name = self.nameField.text;
person.age = [self.ageField.text integerValue];
[NSKeyedArchiver archiveRootObject:person toFile:file];
如果想解档需要调用[NSKeyedUnarchiver unarchiveObjectWithFile:path];方法如下
NSString *file = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES).firstObject stringByAppendingPathComponent:@"student.data"];
Person *person = [NSKeyedUnarchiver unarchiveObjectWithFile:file];
if (person) {
self.avatarView.image = person.avatar;
self.nameField.text = person.name;
self.ageField.text = [NSString stringWithFormat:@"%ld", person.age];
}
在这之前还是要定义属性和实现协议方法
//1.遵循NSCoding协议
@interface student : NSObject //2.设置属性
@property (strong, nonatomic) UIImage *avatar;
@property (copy, nonatomic) NSString *name;
@property (assign, nonatomic) NSInteger age;
@end
//解档
- (id)initWithCoder:(NSCoder *)aDecoder {
if ([super init]) {
self.avatar = [aDecoder decodeObjectForKey:@"avatar"];
self.name = [aDecoder decodeObjectForKey:@"name"];
self.age = [aDecoder decodeIntegerForKey:@"age"];
}
return self;
}
//归档
- (void)encodeWithCoder:(NSCoder *)aCoder {
[aCoder encodeObject:self.avatar forKey:@"avatar"];
[aCoder encodeObject:self.name forKey:@"name"];
[aCoder encodeInteger:self.age forKey:@"age"];
}
2.4 SQLite
以下并不是推广博客,主要是很多内容,没有必要写第二遍,即使写,也没用专门一篇详细。
关于SQLite讲解,请查看以前博客https://www.cnblogs.com/guohai-stronger/p/9218175.html
SQLite的使用还是很麻烦的,在一般的项目开发中,会使用FMDB操作,这个也有讲解,https://www.cnblogs.com/guohai-stronger/p/9246653.html
对于SQLite和FMDB的使用区别,看这篇博客希望对大家有所帮助 https://www.cnblogs.com/guohai-stronger/p/9251131.html
2.5 CoreData
CoreData目前自己所从事的工作还没有使用过CoreData,但自己通过尝试demo,还是有所感悟,关于CoreData的理解,自己也写了一个博客,https://www.cnblogs.com/guohai-stronger/p/9254293.html
以上就是iOS数据存储的5中方式,都是自己真实项目所使用的,希望对大家有所帮助 !!!如有错误的地方请告诉我,大家共同进步。