ios - 使用解码协议(protocol)解析 Json (Codable)

标签 ios swift encode codable decodable

我正在尝试使用 Codable 协议(protocol)解析 JSON 但出现异常。 此异常是由于第二个响应的 nestedkeyedcontainer 中的 null 值引起的。

它对下面的 json 响应工作正常

    let jsonStr = """
   {
     "response": {
         "PURCHASE_FREE_RESPONSE": {
                                "TOPUP": null,
                                "TOTAL_PAYMENT_AMOUNT": "0",
                                "TAX": null,
                                "VAT": null,
                               ........
                                 "CUSTOM_FIELD_2": null,
                                 "CUSTOM_FIELD_3": null,
                                 "FEE": null
                  }
         },
     "API-Code": "ttttt",
     "statuscode": 200,
     "statusMessage": "SUCCESS",
     "respCode": {
     "ERROR_CODE": "0",
     "ERROR_DESC": "Success"
      }
     }
    """
    //=========Second Response===============

        let jsonStr = """
         {
            **"response": null,**
            "API-Code": null,
            "statuscode": 500,
            "statusMessage": "Undefined index: HOUSE_NO",
              **"respCode": null**
           }
 
        """

我的模型看起来像:-

struct SimPurResponse:Decodable{
let topUp:String
let totalPaymentAmt:String
let transId:String


let apiCode:String
let statuscode:Int
let statusMessage:String

let errorCode:String
let errorDesc:String


private enum ResponseOuter : String, CodingKey {
    case response
    case apiCode      = "API-Code"
    case statuscode   = "statuscode"
    case respCode
    case statusMessage = "statusMessage"
}

enum RespCodeKeys : String, CodingKey {
    case errorCode   = "ERROR_CODE"
    case errorDesc   = "ERROR_DESC"
}

private enum Response : String, CodingKey {
    case PURCHASE_FREE_SIM_RESPONSE
}

private enum purchaseFreeSimResInfo : String, CodingKey {
    case topUp            = "TOPUP"
    case totalPaymentAmt  = "TOTAL_PAYMENT_AMOUNT"
    case transId          = "TRANSACTION_ID"
    case tax              = "TAX"
    case vat              =  "VAT"
    case shipingAmt       = "SHIPPING_AMOUNT"
    
}


init(from decoder :Decoder) throws {
    let container     = try decoder.container(keyedBy: ResponseOuter.self)
    self.apiCode      = try container.decodeIfPresent(String.self, forKey: .apiCode) ?? ""
    self.statuscode    = try container.decodeIfPresent(Int.self, forKey: .statuscode) ?? 600
    self.statusMessage = try container.decodeIfPresent(String.self, forKey: .statusMessage) ?? ""
    
    let respCodeCont =  try container.nestedContainer(keyedBy: RespCodeKeys.self, forKey: .respCode)
    self.errorCode   =  try respCodeCont.decodeIfPresent(String.self, forKey: .errorCode) ?? ""
    self.errorDesc   =  try respCodeCont.decodeIfPresent(String.self, forKey: .errorDesc) ?? ""
    
   
    let purcCont          = try container.nestedContainer(keyedBy: Response.self, forKey: .response)
    let purSimCont        = try purcCont.nestedContainer(keyedBy: purchaseFreeSimResInfo.self, forKey: .PURCHASE_FREE_SIM_RESPONSE)

      self.topUp            = try purSimCont?.decodeIfPresent(String.self,  forKey: .topUp) ?? ""
     self.totalPaymentAmt    = try purSimCont?.decodeIfPresent(String.self, forKey: .totalPaymentAmt) ?? ""
     self.transId           = try purSimCont?.decodeIfPresent(String.self, forKey: .transId) ?? ""
     self.tax              = try purSimCont?.decodeIfPresent(String.self, forKey: .tax) ?? ""
     self.vat               = try purSimCont?.decodeIfPresent(String.self, forKey: .vat) ?? ""
     self.shipingAmt       = try purSimCont?.decodeIfPresent(String.self, forKey: .shipingAmt) ?? ""
    
}

        let dataJosn = Data(jsonStr.utf8)

 do {
     let simDetail = try JSONDecoder().decode(SimPurResponse.self, from: dataJosn)

   print(simDetail)


    }catch let error {
      print(error)
    }

我在处理这两种情况时遇到了问题。以上完整代码

最佳答案

只是将“尝试”更改为“尝试”?在解码器初始化 block 中访问 nestedContainer 时,并在需要的地方添加可选链接:

init(from decoder :Decoder) throws {

    let data = try decoder.container(keyedBy: ResponseOuter.self)

    self.apiCode      = try data.decodeIfPresent(String.self, forKey: .apiCode) ?? ""
    self.statuscode    = try data.decodeIfPresent(Int.self, forKey: .statuscode) ?? 600
    self.statusMessage = try data.decodeIfPresent(String.self, forKey: .statusMessage) ?? ""

    let respCodeCont =  try? data.nestedContainer(keyedBy: RespCodeKeys.self, forKey: .respCode)
    self.errorCode   =  try respCodeCont?.decodeIfPresent(String.self, forKey: .errorCode) ?? ""
    self.errorDesc   =  try respCodeCont?.decodeIfPresent(String.self, forKey: .errorDesc) ?? ""

    let purcCont          = try? data.nestedContainer(keyedBy: Response.self, forKey: .response)
    let purSimCont        = try purcCont?.nestedContainer(keyedBy: purchaseFreeSimResInfo.self, forKey: .PURCHASE_FREE_SIM_RESPONSE)
    self.topUp            = try purSimCont?.decodeIfPresent(String.self, forKey: .topUp) ?? ""
    self.totalPaymentAmt    = try purSimCont?.decodeIfPresent(String.self, forKey: .totalPaymentAmt) ?? ""
    self.transId           = try purSimCont?.decodeIfPresent(String.self, forKey: .transId) ?? ""
    self.tax              = try purSimCont?.decodeIfPresent(String.self, forKey: .tax) ?? ""
    self.vat               = try purSimCont?.decodeIfPresent(String.self, forKey: .vat) ?? ""
    self.shipingAmt       = try purSimCont?.decodeIfPresent(String.self, forKey: .shipingAmt) ?? ""

    self.itgTransId          = try purSimCont?.decodeIfPresent(String.self, forKey: .itgTransId) ?? ""
    self.chanelTransId       = try purSimCont?.decodeIfPresent(String.self, forKey: .chanelTransId) ?? ""
    self.paymentResponse     = try purSimCont?.decodeIfPresent(String.self, forKey: .paymentResponse) ?? ""
    self.apmResponse        = try purSimCont?.decodeIfPresent(String.self, forKey: .apmResponse) ?? ""
    self.applicationType    = try purSimCont?.decodeIfPresent(String.self, forKey: .applicationType) ?? ""
    self.nameorPageId       = try purSimCont?.decodeIfPresent(String.self, forKey: .nameorPageId) ?? ""
    self.url                = try purSimCont?.decodeIfPresent(String.self, forKey: .url) ?? ""

    self.customField1       = try purSimCont?.decodeIfPresent(String.self, forKey: .customField1) ?? ""
    self.customField2       = try purSimCont?.decodeIfPresent(String.self, forKey: .customField2) ?? ""
    self.customField3       = try purSimCont?.decodeIfPresent(String.self, forKey: .customField3) ?? ""
    self.fee                 = try purSimCont?.decodeIfPresent(String.self, forKey: .fee) ?? ""
}

希望这对您有所帮助。干杯。

关于ios - 使用解码协议(protocol)解析 Json (Codable),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58350722/

相关文章:

ios - 在 Swift 中,如何使用 NSPredicate 在包含字符串和 NSNumber 的自定义类中进行搜索?

swift - 在 Swift 3 中通过 IBInspectable 传递选择器

video - 如果我用相同的比特率连续两次 ffmpeg 编码会发生什么

ios Swift 处理 completionHandler

C# 将具有多种类型字符的字符串编码为 HTML

ios - Base64 图像编码 Swift 4 iOS

ios - Braintree iOS SDK 说没有已知的类方法

iphone - 在没有任何用户名的情况下投票一次

ios - 相同的 View Controller UI,不同的功能

ios - 禁用表格 View 中的弹跳使得在尝试过度滚动后点击单元格两次