iOS中常用的持久化存储方式有好几种

前言

iOS中常用的持久化存储方式有好几种:

  • 偏好设置(NSUserDefaults)
  • plist文件存储
  • 归档
  • SQLite3
  • Core Data

这里不细讲数据库,只针对性地讲讲文件存储、归档/解档、偏好设置等。

在此之前,我们需要先讲讲沙盒(Sandbox)才能继续讲解。

沙盒

每个iOS应用都有自己的应用沙盒(应用沙盒就是文件系统目录),与其他文件系统隔离。应用必须待在自己的沙盒里,其他应用不能访问该沙盒。沙盒下的目录如下:

  • Application:存放程序源文件,上架前经过数字签名,上架后不可修改
  • Documents: 保存应⽤运行时生成的需要持久化的数据,iTunes同步设备时会备份该目录。例如,游戏应用可将游戏存档保存在该目录
  • tmp: 保存应⽤运行时所需的临时数据,使⽤完毕后再将相应的文件从该目录删除。应用 没有运行时,系统也可能会清除该目录下的文件。iTunes同步设备时不会备份该目录。
  • Library/Caches: 保存应用运行时⽣成的需要持久化的数据,iTunes同步设备时不会备份 该目录。⼀一般存储体积大、不需要备份的非重要数据,比如网络数据缓存存储到Caches下
  • Library/Preference: 保存应用的所有偏好设置,如iOS的Settings(设置) 应⽤会在该目录中查找应⽤的设置信息。iTunes同步设备时会备份该目录

NSUserDefaults

NSUserDefaults是个单例类,用于存储少量数据。NSUserDefaults实际上对plist文件操作的封装,更方便我们直接操作,一般用于存储系统级别的偏好设置。比如我们经常将登录后的用户的一些设置通过NSUserDefaults存储到plist文件中。

有很多App,他们也是将用户的账号和密码存储在偏好设置中。我们不讲安全性问题,因此不讨论存储在偏好设置下是否安全。

使用起来非常简单,如下:

1
2
3
4
5
6
7
8
9
10
11
12
 
// 写入文件
- ( void ) saveUserName : ( NSString * ) userName password : ( NSString * ) password {
   [ [ NSUserDefaults standardUserDefaults ] setObject :userName forKey : @"username" ] ;
   [ [ NSUserDefaults standardUserDefaults ] setObject :password forKey : @"password" ] ;
   [ [ NSUserDefaults standardUserDefaults ] synchronize ] ;
}
 
// 在用的时候,就可以读取出来使用
NSString *userName = [ [ NSUserDefaults standardUserDefaults ] objectForKey : @"username" ] ;
NSString *password = [ [ NSUserDefaults standardUserDefaults ] objectForKey : @"password" ] ;
 

存储到偏好设置的只有系统已经提供好的类型,比如基本类型、NSNumber、NSDictionary、NSArray等。对于NSObject及继承于NSObject的类型,是不支持的。如下:

1
2
3
4
5
6
7
 
NSObject *obj = [ [ NSObject alloc ] init ] ;
[ [ NSUserDefaults standardUserDefaults ] setObject :obj forKey : @"obj" ] ;
 
// 就会崩溃
Terminating app due to uncaught exception 'NSInvalidArgumentException' , reason : 'Attempt to insert non-property list object <NSObject: 0x7fb502680cb0> for key obj'
 

plist存储

有的时候,我们需要将下载的数据存储到文件中存储起来,比如,有时候我们将下载起来的城市的数据存储到本地,当更新城市的顺序时,下次也能够按照最后一次操作的顺序来显示出来。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
 
// 数据存储,是保存到手机里面,
// Plist存储,就是把某些数据写到plist文件中
// plist存储一般用来存储数组和字典
// Plist存储是苹果特有,只有苹果才能生成plist
// plist不能存储自定义对象,如NSObject、model等
NSDictionary *dict = @ { @"age" : @"18" , @"name" : @"USER" } ;
  
// 保存应用沙盒(app安装到手机上的文件夹)
// Caches文件夹
// 在某个范围内容搜索文件夹的路径
// directory:获取哪个文件夹
// domainMask:在哪个范围下获取 NSUserDomainMask:在用户的范围内搜索
// expandTilde是否展开全路径,YES:展开
NSString *cachePath = NSSearchPathForDirectoriesInDomains ( NSCachesDirectory , NSUserDomainMask , YES ) [ 0 ] ;
NSLog ( @"%@" , cachePath ) ;
  
// 拼接文件路径
NSString *filePath = [ cachePath stringByAppendingPathComponent : @"data.plist" ] ;
  
// 获取应用沙盒
NSString *homePath = NSHomeDirectory ( ) ;
NSLog ( @"%@" , homePath ) ;
  
// File:文件全路径 => 所有文件夹路径 + 文件路径
[ dict writeToFile :filePath atomically :YES ] ;
// 将数据取出来
NSLog ( @"%@" , [ NSDictionary dictionaryWithContentsOfFile :filePath ] ) ;
 

我们看看打印的结果:

1
2
3
4
5
6
7
8
 
2016 - 02 - 17 22 : 14 : 43.055 iOSPersistentStorageDemo [ 25471 : 809758 ] / Users / huangyibiao / Library / Developer / CoreSimulator / Devices / CF3A5A4C - 486F - 4A72 - 957B - 2AD94BD90EC1 / data / Containers / Data / Application / 65E8F814 - 45E5 - 420C - A174 - 822A7830748E / Library / Caches
2016 - 02 - 17 22 : 14 : 43.055 iOSPersistentStorageDemo [ 25471 : 809758 ] / Users / huangyibiao / Library / Developer / CoreSimulator / Devices / CF3A5A4C - 486F - 4A72 - 957B - 2AD94BD90EC1 / data / Containers / Data / Application / 65E8F814 - 45E5 - 420C - A174 - 822A7830748E
2016 - 02 - 17 22 : 14 : 43.056 iOSPersistentStorageDemo [ 25471 : 809758 ] {
     age = 18 ;
     name = USER ;
}
 

注意:操作plist文件时,文件路径一定要是全路径。

归档(NSKeyedArchiver)

自定义对象应用范围很广,因为它对应着MVC中的Model层,即实体类。在程序中,我们会在Model层定义很多的entity,例如User、Teacher、Person等。

那么对自定义对象的归档显得重要的多,因为很多情况下我们需要在Home键之后保存数据,在程序恢复时重新加载,那么,归档便是一个好的选择。

下面我们自定义一个Person类:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
 
// 要使对象可以归档,必须遵守NSCoding协议
@interface Person : NSObject <NSCoding>
 
@property ( nonatomic , assign ) int age ;
@property ( nonatomic , strong ) NSString *name ;
 
@end
 
@implementation Person
 
// 什么时候调用:只要一个自定义对象归档的时候就会调用
- ( void ) encodeWithCoder : ( NSCoder * ) aCoder {
     [ aCoder encodeObject :self . name forKey : @"name" ] ;
     [ aCoder encodeInt :self . age forKey : @"age" ] ;
}
 
- ( id ) initWithCoder : ( NSCoder * ) aDecoder {
     if ( self = [ super init ] ) {
       self . name = [ aDecoder decodeObjectForKey : @"name" ] ;
       self . age = [ aDecoder decodeIntForKey : @"age" ] ;
     }
     return self ;
}
@end
 

如何将自定义对象归档和解档:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
 
- ( void ) savePerson {
     // 归档:plist存储不能存储自定义对象,此时可以使用归档来完成
     Person *person = [ [ Person alloc ] init ] ;
     person . age = 18 ;
     person . name = @"USER" ;
        
     // 获取tmp目录路径
     NSString *tempPath = NSTemporaryDirectory ( ) ;
        
     // 拼接文件名
     NSString *filePath = [ tempPath stringByAppendingPathComponent : @"person.data" ] ;
        
     // 归档
     [ NSKeyedArchiver archiveRootObject :person toFile :filePath ] ;
}
 
- ( void ) readPerson {
     // 获取tmp  
     NSString *tempPath = NSTemporaryDirectory ( ) ;
        
     // 拼接文件名
     NSString *filePath = [ tempPath stringByAppendingPathComponent : @"person.data" ] ;
        
     // 解档
     Person *p = [ NSKeyedUnarchiver unarchiveObjectWithFile :filePath ] ;
     NSLog ( @"%@ %d" , p . name , p . age ) ;
}
 

假设我们定义了一个自定义的view,这个view是用xib或者storybard来生成的,那么我们我一定如下方法时,就需要如下实现:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
 
@implementation CustomView
 
// 解析xib,storyboard文件时会调用
- ( id ) initWithCoder : ( NSCoder * ) aDecoder {
     // 什么时候调用[super initWithCoder:aDecoder]?
     // 只要父类遵守了NSCoding协议,就调用[super initWithCoder:aDecoder]
     if ( self = [ super initWithCoder :aDecoder ] ) {
         NSLog ( @"%s" , __func__ ) ;
     }
    
     return    self ;
}
 
@end
 

如果想要学习如何通过runtime来实现自动归档、解档,请阅读文章: 通过runtime自动归档/解档
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值