Swift 4 个可解码嵌套 json 具有随机密钥属性

Swift 4 decodable nested json with random key attributes

我在解码时遇到问题 json。我遵循了很多教程,但没有使用复杂的 json 结构。为简单起见,我将代码最小化并以 Dog 为例。

在接下来的 json 中,我主要只对 Dog 结构感兴趣。 json "Data" 属性包含随机狗名。所以我不能使用编码键,因为我不知道属性名称。

{
     "Response": "success"
     "BaseLinkUrl": "https://wwww.example.com",
     "Data": {
         "Max": {
             "name": "Max",
             "breed": "Labrador"
         },
         "Rocky": {
             "name": "Rocky",
             "breed": "Labrador"
         },
         ...
    }
}

我有以下结构:

struct DogResponse : Decodable {
    let data : DogResponseData

    enum CodingKeys: String, CodingKey {
        case data = "Data"
    }
}

struct DogResponseData: Decodable {
    let dog: Dog //this is a random variable name

    enum CodingKeys: String, CodingKey {
        case dog = "??random_variable_dog_name??"
    }
}

struct Dog: Decodable {
    let name: String
    let type: String

    enum CodingKeys: String, CodingKey {
        case name
        case type = "breed"
    }
}

正在收集 Dog 结构:

let dogResponse = try JSONDecoder().decode(DogResponse.self, from: data)
print(dogResponse)

我需要在我的 "DogResponseData" 结构中做什么才能让 swift 识别包含我的 Dog 结构的随机变量?

一个可能的解决方案是编写一个自定义初始化程序来将字典解码为 [String:Dog] 并将值映射到数组

struct Dog : Decodable {
    let name : String
    let breed : String
}

struct DogResponse : Decodable {
    let dogs : [Dog]

    private enum CodingKeys: String, CodingKey {
        case data = "Data"
    }

    init(from decoder: Decoder) throws {
        let values = try decoder.container(keyedBy: CodingKeys.self)
        let data = try values.decode([String : Dog].self, forKey: .data)
        dogs = Array(data.values)
    }
}

let dogResponse = try JSONDecoder().decode(DogResponse.self, from: data)
print(dogResponse.dogs)

============================================= ==============================

或者如果你想保留字典结构,它仍然更短

struct Dog : Decodable {
    let name : String
    let breed : String
}

struct DogResponse : Decodable {
    let dogs : [String : Dog]

    private enum CodingKeys: String, CodingKey {
        case dogs = "Data"
    }
}

请记住,CodingKey 是一个协议,不一定是 enum。所以你可以把它变成 struct 并且它会接受你扔给它的任何随机字符串值。