如何在“可编码”响应中使用字典?

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&lt;String, Any&gt; 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 - 我将如何编码?我正在搜索示例,但没有什么真正适合我的情况。??谢谢

以上是关于如何在“可编码”响应中使用字典?的主要内容,如果未能解决你的问题,请参考以下文章

如何在可编码结构中使用计算属性(swift)

使用泛型/可编码 w/ API 响应 204 NO CONTENT

多部分请求 - 带有 UIImage 的可编码结构

Swift 4 可编码;如何使用单个根级密钥解码对象

带有可编码数组的 swift 5 抽象网络响应

如何在目标 c 数据模型类中使用 Codable 协议?