Чтение файла JSON с помощью Swift 3

У меня есть файл JSON, называемый point.json, и функция чтения вроде:

private func readJson() {
    let file = Bundle.main.path(forResource: "points", ofType: "json")
    let data = try? Data(contentsOf: URL(fileURLWithPath: file!))
    let jsonData = try? JSONSerialization.jsonObject(with: data!, options: []) as! [String:Any]
    print(jsonData)
}

Это не работает, любая помощь?

Ответ 1

Ваша проблема заключается в том, что вы принудительно разворачиваете значения, а в случае ошибки вы не можете знать, откуда она взялась.

Вместо этого вы должны обрабатывать ошибки и безопасно разворачивать свои опции.

И как @vadian правильно отмечает в своем комментарии, вы должны использовать Bundle.main.url.

private func readJson() {
    do {
        if let file = Bundle.main.url(forResource: "points", withExtension: "json") {
            let data = try Data(contentsOf: file)
            let json = try JSONSerialization.jsonObject(with: data, options: [])
            if let object = json as? [String: Any] {
                // json is a dictionary
                print(object)
            } else if let object = json as? [Any] {
                // json is an array
                print(object)
            } else {
                print("JSON is invalid")
            }
        } else {
            print("no file")
        }
    } catch {
        print(error.localizedDescription)
    }
}

При кодировании в Swift обычно ! является запахом кода. Конечно, есть исключения (IBOutlets и другие), но старайтесь не использовать силовую развертку с ! самостоятельно и всегда разворачивать безопасно.

Ответ 2

Приведенный ниже код Swift 5/iOS 12.3 показывает возможное переписывание вашего метода, позволяющее избежать принудительного развертывания необязательных значений и аккуратно обрабатывать возможные ошибки:

import Foundation

func readJson() {
    // Get url for file
    guard let fileUrl = Bundle.main.url(forResource: "Data", withExtension: "json") else {
        print("File could not be located at the given url")
        return
    }

    do {
        // Get data from file
        let data = try Data(contentsOf: fileUrl)

        // Decode data to a Dictionary<String, Any> object
        guard let dictionary = try JSONSerialization.jsonObject(with: data, options: []) as? [String: Any] else {
            print("Could not cast JSON content as a Dictionary<String, Any>")
            return
        }

        // Print result
        print(dictionary)
    } catch {
        // Print error if something went wrong
        print("Error: \(error)")
    }
}