首页 > 解决方案 > Swift 4 Decodable:解码复杂的 JSON

问题描述

我必须解码字典数组,其中键是枚举,值是模型对象。

这是我的示例 JSON,

[
  {
    "nanomp4": {
      "url": "https://media.tenor.com/videos/a1da4dcf693c2187615721d866decf00/mp4", 
      "dims": [
        150, 
        138
      ], 
      "duration": 2.0, 
      "preview": "https://media.tenor.com/images/17d523e6b7c3c9a4ca64566a1890d94d/tenor.png", 
      "size": 70381
    }, 
    "nanowebm": {
      "url": "https://media.tenor.com/videos/aa983425114e32ab446f669d91611938/webm", 
      "dims": [
        150, 
        138
      ], 
      "preview": "https://media.tenor.com/images/17d523e6b7c3c9a4ca64566a1890d94d/tenor.png", 
      "size": 53888
    }, 
  },
  {
    "nanomp4": {
    "url": "https://media.tenor.com/videos/a1da4dcf693c2187615721d866decf00/mp4",
    "dims": [
          150,
          138
          ],
    "duration": 2.0,
    "preview": "https://media.tenor.com/images/17d523e6b7c3c9a4ca64566a1890d94d/tenor.png",
    "size": 70381
    },
  }
]

这是我的解码代码,

do {
    let data = try Data(contentsOf: fileURL)
    let decoder = JSONDecoder()
    let collection = try decoder.decode([[GIFFormat:Media]].self, from: data)

    print(collection)
} catch {
    print("Error in parsing/decoding JSON: \(error)")
}

GIFFormat是 Enum &Media是模型对象,它们解码得非常好。

enum GIFFormat: String, Decodable {

    case nanoMP4    = "nanomp4"
    case nanoWebM   = "nanowebm"
}

struct Media: Decodable {
    let url: URL?        
    let dims: [Int]?
    let duration: Double?
    let preview: URL?
    let size: Int64?
}

我的控制台打印,

typeMismatch(Swift.Array<Any>, Swift.DecodingError.Context(codingPath: [_JSONKey(stringValue: "Index 0", intValue: 0)], debugDescription: "Expected to decode Array<Any> but found a dictionary instead.", underlyingError: nil))

谁能向我解释这里到底出了什么问题?

标签: iosiphoneswiftcodabledecodable

解决方案


即使rawValueforGIFFormatStringGIFFormat它本身也是一个枚举。你应该更新

let collection = try decoder.decode([[GIFFormat:Media]].self, from: data)

let collection = try decoder.decode([[GIFFormat.RawValue:Media]].self, from: data)

更新:回应您的评论

现在要访问值,我需要像这样使用 collection?.first?[GIFFormat.mp4.rawValue]?.url. 这又是丑陋的!

我建议进行一些重构。您可以从enum完全删除您的开始。保持你的Media结构。创建一个新Collection结构

struct Collection: Decodable {
    let nanomp4: Media!
    let nanowebm: Media!
}

然后,您可以将上面的行更新为

let collection = try decoder.decode([Collection].self, from: data)

丑陋的线条变成

collection.first?.nanomp4.url

注意:此解决方案假定您只有nanomp4&nanowebm作为枚举值。如果不是这种情况,那么这可能不是最好的解决方案,您可能必须使用第一个解决方案。


推荐阅读