从 .plist 读取总是 returns nil

reading from .plist always returns nil

我从 plist 读取的所有尝试都返回了 nil 值,我在 Xcode 6 和 Xcode beta 7。另外,堆栈上有很多类似的问题,我已经尝试了很多,但是 none 其中解决了这个问题。

我已通过单击添加了我的 words.plist

{我的项目} > 目标 > 构建阶段 > 复制捆绑资源

然后我在 ViewController 中尝试了以下代码的几种变体:

override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view, typically from a nib.
    let paths = NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true) as NSArray
    let documentsDirectory = paths[0] as! String
    let path = documentsDirectory.stringByAppendingPathComponent("words.plist")

    let fileManager = NSFileManager.defaultManager()

    //check if file exists
    if(!fileManager.fileExistsAtPath(path)) {
        // If it doesn't, copy it from the default file in the Bundle
        if let bundlePath = NSBundle.mainBundle().pathForResource("words", ofType: "plist") {

            let resultDictionary = NSMutableDictionary(contentsOfFile: bundlePath)
            println("Bundle words file is --> \(resultDictionary?.description)") // this is nil!!!

            fileManager.copyItemAtPath(bundlePath, toPath: path, error: nil)

        } else {
            println("words not found. Please, make sure it is part of the bundle.")
        }
    } else {
        println("words already exits at path.")
        // use this to delete file from documents directory
        //fileManager.removeItemAtPath(path, error: nil)

    }
    print("entering if-let")
    if let pfr = NSBundle.mainBundle().pathForResource("words", ofType: "plist") {
        print("\nin let\n")
        print(pfr)
        print("\nentering dict if-let\n")
        if let dict = NSDictionary(contentsOfFile: path) as? Dictionary<String, AnyObject> {
            // use swift dictionary as normal
            print("\nin let\n")
            print(dict)
        }   
    }
}

问题

为什么我得到一个 nil 值,添加 plist 文件并从中读取的正确方法是什么?


更新:

在我的 if 语句中,以下是 nil:

let resultDictionary = NSMutableDictionary(contentsOfFile: bundlePath)
println("Bundle words file is --> \(resultDictionary?.description)") // this is nil!!!

对我来说,这表明 Xcode 不知道我的 words.plist 文件,或者我将 bundlePath 指向了错误的位置。

问题:

如评论中@Steven Fisher所述。我的 .plist 文件是 Array 而不是 NSDictionary。所以我只需要从我的代码中切换两行:

let resultDictionary = NSMutableDictionary(contentsOfFile: bundlePath)

let resultDictionary = NSMutableArray(contentsOfFile: bundlePath)

还有

if let dict = NSDictionary(contentsOfFile: path) { //...

if let dict = NSArray(contentsOfFile: path) { //..

最终工作代码:

override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view, typically from a nib.
        let paths = NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true) as NSArray
        let documentsDirectory = paths[0] as! String
        let path = documentsDirectory.stringByAppendingPathComponent("words.plist")

        let fileManager = NSFileManager.defaultManager()

        //check if file exists
        if(!fileManager.fileExistsAtPath(path)) {
            // If it doesn't, copy it from the default file in the Bundle
            if let bundlePath = NSBundle.mainBundle().pathForResource("words", ofType: "plist") {

                let resultDictionary = NSMutableArray(contentsOfFile: bundlePath)
                println("Bundle words file is --> \(resultDictionary?.description)")

                fileManager.copyItemAtPath(bundlePath, toPath: path, error: nil)

            } else {
                println("words not found. Please, make sure it is part of the bundle.")
            }
        } else {
            println("words already exits at path.")
            // use this to delete file from documents directory
            //fileManager.removeItemAtPath(path, error: nil)

        }
        print("entering if-let")
        if let pfr = NSBundle.mainBundle().pathForResource("words", ofType: "plist") {
            print("\nin let\n")
            print(pfr)
            print("\nentering dict if-let\n")
            if let dict = NSArray(contentsOfFile: pfr) {
                // use swift dictionary as normal
                print("\nin let\n")
                print(dict)
            }

        }
    }