如何在“可编码”响应中使用字典?
Posted
技术标签:
【中文标题】如何在“可编码”响应中使用字典?【英文标题】:How to use a dictionary in `codable` response? 【发布时间】:2017-11-06 21:09:16 【问题描述】:我收到了来自 JSON 查询的以下响应。如何将字典表示为可编码?我已缩短 JSON 响应以节省空间。
"result":[
"delivery_address": "",
"made_sla": "true",
"watch_list": "",
"upon_reject": "cancel",
"location":
"link": "https://foo.com/api/now/table/cmn_location/753ac76f4fd9320066bfb63ca310c79b",
"value": "753ac76f4fd9320066bfb63ca310c79b"
]
struct ResultList : Codable
let result: [Result]
struct Result : Codable
let delivery_address: String
let made_sla: String
let watch_list: String
let upon_reject: String
let location: Location
struct Location: Codable
let link: String?
let value: String?
let decoder = JSONDecoder()
do
let todo = try decoder.decode(ResultList.self, from: responseData)
print("todo \(todo)")
catch
print("error trying to convert data to JSON")
print(error)
我收到以下错误:
"Expected to decode Dictionary<String, Any> but found a string/data instead.", underlyingError: nil))
【问题讨论】:
那个json不会给你那个错误 那不是真正的代码/JSON。Index 2
来自哪里?并且至少 JSON 必须包裹在
完整的 JSON 响应很长,我不想完整地发布它。但错误表明 .location "Expected to decode Dictionary<String, Any> but found a string/data instead."
正如@dan 提到的,这个 JSON(包装在
中)确实有效。
更新了 JSON 响应,并添加了一些字段。
【参考方案1】:
基于所有的 cmets,我相信你实际解码的 JSON 看起来更像这样:
"result": [
"delivery_address": "",
"made_sla": "true",
"watch_list": "",
"upon_reject": "cancel",
"location":
"link": "https://foo.com/api/now/table/cmn_location/753ac76f4fd9320066bfb63ca310c79b",
"value": "753ac76f4fd9320066bfb63ca310c79b"
,
"delivery_address": "",
"made_sla": "true",
"watch_list": "",
"upon_reject": "cancel",
"location": ""
]
所以有些记录有位置,有些记录将位置编码为空字符串。基本上,这在很多层面上都把 JSON 搞砸了,无论生成什么代码都应该修复。坏消息,我相信这不会发生。好消息是我们至少可以在本地修复它。
我们将不得不手动解码,所以我们不妨趁着这里把剩下的烂摊子清理干净。首先是名称不符合 Swift 命名约定。我们可以使用CodingKeys
解决这个问题。我们还可以为当前输入错误的字符串提供真实类型(Bool 和 Rejection 枚举)。
enum Rejection: String, Codable
case cancel
struct Result : Codable
let deliveryAddress: String
let madeSLA: Bool
let watchList: String
let uponReject: Rejection
let location: Location?
private enum CodingKeys: String, CodingKey
case deliveryAddress = "delivery_address"
case madeSLA = "made_sla"
case watchList = "watch_list"
case uponReject = "upon_reject"
case location
现在我们只需要能够解码它。请注意,我将 Location 设为可选。显然它有时不存在,因此您要么需要一个默认值,要么它需要是可选的。我选择了后者。解码所有这些非常简单:
init(from decoder: Decoder) throws
let container = try decoder.container(keyedBy: CodingKeys.self)
deliveryAddress = try container.decode(String.self, forKey: .deliveryAddress)
madeSLA = try container.decode(String.self, forKey: .madeSLA) == "true"
watchList = try container.decode(String.self, forKey: .watchList)
uponReject = try container.decode(Rejection.self, forKey: .uponReject)
location = try? container.decode(Location.self, forKey: .location)
最后一行是您的实际问题。它只是说如果我们不能将其解码为Location
,请将其设置为 nil。我们这里可以更严格一些,先解码为Location,再解码为String,再检查String是否为空,但这里用nil表示解码失败感觉合理。
【讨论】:
感谢您的详细回复。它工作得很好。感谢所有其他帮助达到这一点的人。我希望这对其他人有帮助。【参考方案2】:假设您的 JSON 是(请注意缺少的右括号)
"result": [
"delivery_address": "",
"made_sla": "true",
"watch_list": "",
"upon_reject": "cancel",
"location":
"link": "https://blah/blah/foo",
"value": "fsfdfr32r34rwfwffas"
]
你可以解码这些结构
struct Root : Decodable
let result : [Result]
struct Result : Decodable
let location: Location
struct Location: Decodable
let link: String
let value: String
与
JSONDecoder().decode(Root.self, from: data)
【讨论】:
我正在按照您的建议进行操作,请参阅更新。 然后——再一次——JSON 不是问题中的 JSON(分别在我的回答中)。我的答案中的 JSON / 代码有效。 我可以确认您是正确的。我找到了探针。location
的一些数据不是字典,而是 location: ""
我只需要弄清楚如何检查。
那太恶心了。这意味着您需要手动实现init(from: Decoder)
并在解码location
值时添加一些回退行为。
@JonShier - 我将如何编码?我正在搜索示例,但没有什么真正适合我的情况。??谢谢以上是关于如何在“可编码”响应中使用字典?的主要内容,如果未能解决你的问题,请参考以下文章