Разбор JSON в Swift 3 после загрузки файла JSON, NSCocoaErrorDomain Code=3840
Поэтому я продолжаю получать сообщение об ошибке ниже, когда пытаюсь выполнить JSONSerialization для объекта JSON, который я загружаю для создаваемого приложения. По сути, я создал файл JSON, сохранил его в SugarSync и получил прямую ссылку для загрузки и использования в моей программе. Вроде мой собственный бутлег API.
Я знаю, что он загружается, потому что, если я преобразую файл в строку, используя URL, он распечатывает файл. Проблема в том, что я не могу обойти эту ошибку.
Ошибка домена =NSCocoaErrorDomain Code=3840 "Нет строкового ключа для значения в объекте вокруг символа 2".
Это пример того, как файл читается (очень просто!):
{"buddy": "pal",
"brother": "bear"}
Я попытался перейти на NSData вместо Data, но это создает больше проблем. Попытка сделать NSDictionary в строке JSONSerialization, но это породило больше ошибок. Я также попробовал.mutableContainers, также безрезультатно.
Ниже приведен весь код, включая URL-адрес файла SugarSync, чтобы вы самостоятельно воссоздали ошибку. Я не понимаю, что я делаю неправильно, и если кто-то может помочь, это было бы удивительно.
Спасибо!
// Create destination URL
let documentsURL: URL = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).first as URL!
let destinationFileUrl = documentsURL.appendingPathComponent("downloadedFile.json")
// Create URL to the source file you want to download
let fileURL = URL(string: "https://www.sugarsync.com/pf/D7126167_796_275761334?directDownload=true")
let sessionConfig = URLSessionConfiguration.default
let session = URLSession(configuration: sessionConfig)
let request = URLRequest(url: fileURL!)
let task = session.downloadTask(with: request) { (tempLocalUrl, response, error) in
if let tempLocalUrl = tempLocalUrl, error == nil {
// Success
if let statusCode = (response as? HTTPURLResponse)?.statusCode {
print("Successfully downloaded. Status code: \(statusCode)")
}
do {
print("move files")
try FileManager.default.copyItem(at: tempLocalUrl, to: destinationFileUrl)
let fileData: Data = try Data(contentsOf: destinationFileUrl, options: Data.ReadingOptions.mappedIfSafe)
print(fileData)
let myJson = try JSONSerialization.jsonObject(with: fileData, options: []) as? [[String: Any]]
print(myJson!)
do {
// This converts the file into String and is printed (doesn't get to this because of the error, but it works!)
let jsonText = try String(contentsOf: tempLocalUrl, encoding: String.Encoding.utf8)
print(jsonText)
} catch {
print("failed to print json file")
}
} catch (let writeError) {
print("Error creating a file \(destinationFileUrl) : \(writeError)")
}
} else {
print("Error took place while downloading file. Error description: %@", error?.localizedDescription ?? "unknown")
}
}
task.resume()
РЕДАКТИРОВАТЬ --------------------------- Похоже, сериализация очень чувствительна. Для работы нужно что-то вроде этого:
{"widget": {
"debug": "on",
"window": {
"title": "Sample Konfabulator Widget",
"name": "main_window",
"width": 500,
"height": 500
},
"image": {
"src": "Images/Sun.png",
"name": "sun1",
"hOffset": 250,
"vOffset": 250,
"alignment": "center"
},
"text": {
"data": "Click Here",
"size": 36,
"style": "bold",
"name": "text1",
"hOffset": 250,
"vOffset": 100,
"alignment": "center",
"onMouseUp": "sun1.opacity = (sun1.opacity / 100) * 90;"
}
}}
Но этот не будет работать:
{“website”: {
“iPhone”: {
“blogURL”: ”string”,
“blogHead”: “string”,
“reportURL”: “string”,
“reportHead”: “string”
}
}}
Примечание. Этот веб-сайт не отображает отступы JSON должным образом. Здесь я получаю примеры JSON.
2 ответа
Это сработало в конце. Оказывается, как я уже говорил в моей редакции, сериализатор json очень чувствителен. Поэтому я решил использовать этот онлайн-редактор JSON, чтобы убедиться, что все, что мне нужно, абсолютно правильно, включая правильный шрифт, размер и т. Д.
Вы пытаетесь проанализировать JSON, который имеет тип [[String: Any]]
когда JSON просто словарь ([String: Any]
) не массив словарей.
Попробуйте перейти на это:
let myJson = try JSONSerialization.jsonObject(with: fileData, options: []) as? [String: Any]