Swift 解码 [字符串:任意]

Swift Decode [String: Any]

所以我有这个 API returns [String: Any] 的字典,我知道 Any 是 Decodable 或 [=14= 的数组] 然而我终其一生都无法弄清楚如何获取该字典并将其解码为某种结构:

我的基本上是这样的:

public func call<T: Codable> (completion handler: @escaping (T?) -> ()) {
    let promise = api.getPromise ()
    promise.done (on: DispatchQueue.main, { (results: [String:Any])
        let decodedResults:T? = results.decode (as: T.self) // <-- this is what I want
        handler (decodedResults)
    })
}

我尝试将其转换为数据,然后使用以下方法对其进行解码:

let values = results.compactMap { [=12=].value }
let data = JSONSerialization.data (withJSONObject: values, options: [])
let decodedResult = JSONDecoder().decode(T.self, from: data)

但它总是因 NSInvalidArgumentException 而失败,知道如何解决这个问题吗?

我试图实现但未能实现的另一件事是将值转换为元组,但我发现无法动态创建元组。

解码器将数据转换为可解码值。它们与 [String: Any] 类型或任何其他非数据类型没有任何关系。所以如果你想运行它通过一个解码器,你需要把它转换成JSON编码成Data.

如果 [String: Any] 结果完全是 JSON 序列化安全类型(数组、字典、字符串、数字、null),那么 JSONSerialization.data(withJSONObject:options:) 会让您返回数据,所以你可以重新解码它。你的代码不只是重新编码它的结果,它首先把它变成一个数组:

let values = results.compactMap { [=10=].value }
let data = JSONSerialization.data (withJSONObject: values, options: [])

这很奇怪。您真的是要在这里创建一个数组并丢弃键吗?然后我希望你的 JSONDecoder().decode() 行解码 [T].self 而不是 T.self。所以我期望以下代码(前提是您的 [String: Any] 是 JSON-安全的):

public func call<T: Decodable>(completion handler: @escaping (T?) -> ()) {
    let promise = api.getPromise()
    promise.done(on: .main) { (results: [String:Any]) in
        guard JSONSerialization.isValidJSONObject(results) else {
            handler(nil)
            return
        }

        let data = JSONSerialization.data(withJSONObject: results)
        let decodedResults = try? JSONDecoder().decode(T.self, from: data)
        handler(decodedResults)
    }
}

在评论中,您注意到解码数据([String: Any])不是由基元构成的。在那种情况下,不可能用 JSONSerialization 重新编码它。您需要将 [String: Any] 传递给知道如何处理它的东西。例如:

protocol DictionaryDecodable {
    init?(dictionary: [String: Any])
}

public func call<T: DictionaryDecodable>(completion handler: @escaping (T?) -> ()) {
    let promise = api.getPromise ()
    promise.done(on: .main) { (results: [String:Any])
        handler(T.init(dictionary: results))
    }
}

您的类型将需要实现一个 init?(dictionary:),它可以从 [String: Any].

中解码出它们自己的值