在Swift的DocumentDirectory中存档和取消存档数组

时间:2016-04-16 17:05:17

标签: arrays swift persistence nskeyedarchiver nskeyedunarchiver

我试图通过执行以下代码在持久性内存中保存一个简单的对象数组:

let fileManager=NSFileManager()
     let urls = fileManager.URLsForDirectory(.DocumentDirectory, inDomains: .UserDomainMask)

     if urls.count>0{

         let localDocumentsDirectory=urls[0]
         let archivePath=localDocumentsDirectory.URLByAppendingPathExtension("meditations.archive")
         NSKeyedArchiver.archiveRootObject(self.meditationsArray, toFile: archivePath.path!)
         let restored=NSKeyedUnarchiver.unarchiveObjectWithFile(archivePath.path!)

         print("restored \(restored)")
     }
}

然而,当我按照代码打印恢复日期时,我发现没有。
相反,如果我使用CachesDirectory,阵列很快就会恢复正常,
但是当我重新打开应用程序并尝试加载数据时,它就会丢失。持久保存数据的正确方法是什么?

2 个答案:

答案 0 :(得分:0)

我认为问题在于您应该使用URLByAppendingPathExtension时使用URLByAppendingPathComponent。 “路径扩展名”是文件扩展名,因此您的archivePath是“〜/ Documents.meditations.archive”。它可能暂时与CachesDirectory一起使用,因为它将数据放入某个临时文件中,或者只是从内存中读取它。这应该解决它:

let fileManager = NSFileManager()
let documentDirectoryUrls = fileManager.URLsForDirectory(.DocumentDirectory, .UserDomainMask)

if let documentDirectoryUrl = documentDirectoryUrls.first {
    let fileUrl = documentDirectoryUrl.URLByAppendingPathComponent("meditations.archive")

    // Also, take advantage of archiveRootObject's return value to check if
    // the file was saved successfully, and safely unwrap the `path` property
    // of the URL. That will help you catch any errors.
    if let path = fileUrl.path {
        let success = NSKeyedArchiver.archiveRootObject(meditationArray, toFile: path)

        if !success {
            print("Unable to save array to \(path)")
        }
    } else {
        print("Invalid path")
    }
} else {
    print("Unable to find DocumentDirectory for the specified domain mask.")
}

答案 1 :(得分:0)

我遇到了同样的问题,我无法使用NSKeyedArchiver归档和取消归档对象数组,我认为问题是我正在使用以下方法:

NSKeyedArchiver.archiveRootObject(arrayOfItems, toFile: FileManager.getFileURL("My-File-Name")!)

我认为此方法用于归档对象,而不是对象数组。

无论如何,我通过将整个数组包装在一个对象中找到了解决问题的方法,如下所示:

let myArrayItemsContainer = ArrayItemsContainer()
myArrayItemsContainer.allItems = arrayOfItems
NSKeyedArchiver.archiveRootObject(myArrayItemsContainer, toFile: FileManager.getFileURL("My-File-Name")!)

,然后我使用下面的代码取消归档我的对象:

NSKeyedUnarchiver.unarchiveObject(withFile: FileManager.getFileURL("My-File-Name")!) as? ArrayItemsContainer

我也使用此扩展名来使用FileManager.getFileURL

public extension FileManager {
     /// Returns the URL of the file given a name
     ///
     /// - Parameter fileName: The file name of the file + extension
     /// - Returns: The URL as String
     static func getFileURL(_ fileName: String) -> String? {
     let fileURL = FileManager().urls(for: FileManager.SearchPathDirectory.documentDirectory, in: FileManager.SearchPathDomainMask.userDomainMask).first
         return (fileURL?.appendingPathComponent(fileName).path)
     }
}