在Swift中解码嵌套JSON对象的便捷方法是什么?

时间:2018-01-24 17:03:50

标签: json swift codable decodable

让我们说你有一些JSON:

{
    "status": "error",
    "data": {
        "errormessage": "Could not get user with ID: -1.",
        "errorcode": 14
    }
}

对于给定的Error结构:

struct APIError: Decodable {
    let code: Int?
    let message: String?

    enum CodingKeys: String, CodingKey {
        case code = "errorcode"
        case message = "errormessage"
    }
}

点击Web服务,获取JSON,然后初始化结构:

let urlRequest = URLRequest(url: url)
let session = URLSession.shared
let task = session.dataTask(with: urlRequest)
{ (data, response, error) in
    // Doesn't work because the portion of the JSON we want is in the "data" key
    let e = try? JSONDecoder().decode(APIError.self, from: data)
}
task.resume()

是否有一些简单的方法可以执行data["data"]之类的操作?什么是正确的模型?

解决方案A - 将数据转换为JSON对象,获取我们想要的对象,然后将其转换为Data对象并进行解码。

let jsonFull = (try? JSONSerialization.jsonObject(with: data, options: [])) as? [String: Any]
let json = jsonFull["data"]
let data_error = try? JSONSerialization.data(withJSONObject: json, options: [])
let e = try? JSONDecoder().decode(APIError.self, from: data_error)

解决方案B - 将目标项目包装在另一个结构中

struct temp : Decodable {
    let status: String?
    let data: APIError?
}

let e = try? JSONDecoder().decode(temp.self, from: data).data

解决方案C - 在解码中设置嵌套结构(如果深层有几个对象怎么办?)

let e = try? JSONDecoder().decode([Any, APIError.self], from: data)

我错过了哪些模式?最优雅的方法是什么?

1 个答案:

答案 0 :(得分:0)

您可以使用以下方法:

struct APIError: Decodable {
    let code: Int
    let message: String

    enum CodingKeys: String, CodingKey {
        case data
    }

    enum ErrorCodingKeys: String, CodingKey {
        case code = "errorcode"
        case message = "errormessage"
    }

    init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: CodingKeys.self)
        let nestedContainer = try container.nestedContainer(keyedBy: ErrorCodingKeys.self, forKey: .data)

        code = try nestedContainer.decode(Int.self, forKey: .code)
        message = try nestedContainer.decode(String.self, forKey: .message)
    }
}

let data = try! JSONSerialization.data(withJSONObject: ["status": "error", "data": ["errorcode": 14, "errormessage": "Could not get user with ID: -1."]], options: [])
let error = try! JSONDecoder().decode(APIError.self, from: data)