沙盒
iOS程序默认情况下只能访问程序自己的目录,这个目录被称为“沙盒”。
每个应用程序沙盒都包括以下三个目录:
- Documents:最常用的目录,iTunes同步该应用时会同步此文件夹中的内容,适合存储重要数据。
//获取Documents目录路径的一些代码:
let paths = NSSearchPathForDirectoriesInDomains(
//表明正在查找Documents目录的路径
NSSearchPathDirectory.DocumentDirectory,
//将搜索限制在应用沙盒内
NSSearchPathDomainMask.UserDomainMask,true)
let documentsDirectory = paths[0] as String
//在检索到的路径尾部加一个字符串来创建文件名
let filename = documentsDirectory.stringByAppendingPathComponent("theFile.text")
- Library Caches:保存应用运行时所生成的需要持久化的数据 iTunes同步设备时不会备份 一般存储体积大、不需要备份非重要数据
- Library Preferences:保存应用所有的偏好设置,liOS的Settings(设置)应用会在该目录中查找应用的设置信息,iTunes同步设备时会备份该目录
Library目录获取与Documents类似:
//获取Library目录路径的代码:
let paths = NSSearchPathForDirectoriesInDomains(
NSSearchPathDirectory.LibraryDirectory,
NSSearchPathDomainMask.UserDomainMask,true)
let libraryDirectory = paths[0] as String
- Temp:保存应用运行时所需的临时数据,使用完毕后再将相应的文件从该目录删除。应用没有运行时,系统也可能会清除该目录下的文件。iTunes同步设备时不会备份该目录.
//NSTemporaryDirectory()函数返回的字符串包含应用临时目录的完整路径
let tempPath = NSTemporaryDirectory()
//创建一个会存储在临时目录中的文件
let tempFile = tempPath.stringByAppendingPathComponent("tempFile.text")
属性列表
能将某些特定的类,通过XML文件的方式保存在目录中。
可以被序列化的类型只有如下几种:
Array or NSArray
NSMutableArray
Dictionary or NSDictionary
NSMutableDictionary
NSData
NSMutableData
String or NSString
NSMutableString
NSNumber
NSDate
//写入属性列表
let myArray = [1,2,3]
let array = myArray as NSArray
array.writeToFile("/some/file/location/output.plist",atomically : ture)
注意
- 存储时使用atomically参数,表示是否需要先写入一个辅助文件,再把辅助文件拷贝到目标文件地址。这是为了更安全的写入文件的方法,因为如果应用在保存期间崩溃,则现有文件不会被破坏。尽管增加了一点开销,但是多数情况下还是值得的。
- 属性列表的一个问题就是无法将自定义对象序列化到列表中,另外也不能使用没有在序列化对象类型列表中指定的Cocoa Touch的其他类。这就意味着无法直接使用NSURL、UIImage和UIColor等类。
实践
通过构建一个Dome,在文本框输入数据,应用退出时会将这些字段保存到属性列表文件,然后下次启动时从该文件属性列表文件中加载这些参数。(Dome来自《精通iOS开发(第七版)》省略所有用户界面细节,以及Auto Layout布局)
//查找Documents目录并在其后附加数据文件名
func dataFilePath() -> String {
let paths = NSSearchPathForDirectoriesInDomains(
NSSearchPathDirectory.DocumentDirectory,
NSSearchPathDomainMask.UserDomainMask, true)
let documentsDirectory = paths[0] as NSString
return documentsDirectory.stringByAppendingPathComponent("data.plist") as String
}
在viewDidLoad方法添加以下代码:
let filePath = self.dataFilePath()
//检查数据文件是否存在:不存在就不加载,存在就用该文件的内容实例化数组,并将数组中的对象复制到4个文本框。
if (NSFileManager.defaultManager().fileExistsAtPath(filePath)) {
let array = NSArray(contentsOfFile: filePath) as! [String]
for var i = 0; i < array.count; i++ {
lineFields[i].text = array[i]
}
}
let app = UIApplication.sharedApplication()
//第一个self参数,View Contoller实例会作为观察者接受通知。
//第二个参数将一个选择器传入applicationWillResignActive方法,告诉通知中心在发布该通知后调用这个方法。
//第三个参数UIApplicationWillResignActiveNotification,是接受通知的名称,他是由UIApplication类定义的字符串常量
//第四哥参数app是要从中获取通知的对象
NSNotificationCenter.defaultCenter().addObserver(self,
selector: "applicationWillResignActive:",
name: UIApplicationWillResignActiveNotification,
object: app)
最后创建一个新方法applicationWillResignActive。
//通过调用lineFields数组中每个文本框的text方法构建一个字符串数组,调用valueForKey方法,传递文本text作为参数。
//NSArray类的valueForKey方法实现了迭代获取UITextField实例变量的text值,返回包含这些值的NSString。
func applicationWillResignActive(notification:NSNotification) {
let filePath = self.dataFilePath()
let array = (self.lineFields as NSArray).valueForKey("text") as! NSArray
array.writeToFile(filePath, atomically: true)
}