Bae*_*yun 5 encoding decoding swift
所以我有这个 API 返回一个字典[String: Any],我知道 Any 是Decodable或一个数组Decodable,但是我一生都无法弄清楚如何获取该字典并将其解码为某个结构:
我所拥有的基本上是这样的:
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)
})
}
Run Code Online (Sandbox Code Playgroud)
我尝试将其转换为数据,然后使用以下方法对其进行解码:
let values = results.compactMap { $0.value }
let data = JSONSerialization.data (withJSONObject: values, options: [])
let decodedResult = JSONDecoder().decode(T.self, from: data)
Run Code Online (Sandbox Code Playgroud)
但它总是失败NSInvalidArgumentException,知道如何解决这个问题吗?
我试图实现但未能做到的另一件事是将值转换为元组,但我发现不可能动态创建元组。
解码器将数据转换为可解码值。[String: Any]它们与类型或任何其他非数据类型没有任何关系。所以如果你想通过解码器运行它,你需要将其转换为JSON编码成Data。
如果[String: Any]结果完全是 JSONSerialization 安全类型(数组、字典、字符串、数字、null),那么JSONSerialization.data(withJSONObject:options:)会让您返回数据,以便您可以重新解码它。您的代码不仅仅重新编码其结果,它首先将其转换为数组:
let values = results.compactMap { $0.value }
let data = JSONSerialization.data (withJSONObject: values, options: [])
Run Code Online (Sandbox Code Playgroud)
这很奇怪。你真的想在这里创建一个数组并扔掉键吗?然后我希望你的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)
}
}
Run Code Online (Sandbox Code Playgroud)
在评论中,您注意到解码的数据([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))
}
}
Run Code Online (Sandbox Code Playgroud)
您的类型需要实现一个init?(dictionary:)可以从[String: Any].
| 归档时间: |
|
| 查看次数: |
10621 次 |
| 最近记录: |