首页 > 解决方案 > 使用 Codable 从 2 个不同的 JSON 文件中实例化单个类,而不使用可选项

问题描述

我正在使用提供 2 个 JSON URL 的 API。每个 URL 都包含一个嵌套容器,该容器具有属于同一类和对象的不同属性。

JSON URL 1

{
  "last_updated": 1535936629,
  "xyz": 5,
  "data": {
    "dataList": [
      {
        "id": "42",
        "a1": "a1value",
        "a2": "a2value",
      },
      // ,,,
    ]
  }
}

JSON URL 2

{
  "last_updated": 1536639996,
  "xyz": 5,
  "data": {
    "dataList": [
      {
        "id": "42",
        "lat": "12.345",
        "lon": "67.890",
      },
      // ,,,
    ]
  }
}

我想使用这些 JSON URLS 使用嵌套dataList列表中的项目创建单个 Codable CustomClass 对象,因此我创建了一个Feed结构来处理这两个 JSON 文件。

Feed.swift

import Foundation

Struct Feed: Decodable {
  var lastUpdated: Int
  var xyz: Int
  var data: KeyedDecodingContainer<Feed.dataCodingKey>
  var dataList: [CustomClass]

  enum CodingKeys: String, CodingKey {
    case lastUpdated = "last_updated"
    case xyz
    case data
  }

  enum dataCodingKey: String, CodingKey {
    case dataList
  }

  init(from decoder: Decoder) throws {
    let container = try decoder.container(keyedBy: CodingKeys.self)
    self.lastUpdated = try decoder.decode(Int.self, forKey: .lastUpdated)
    self.xyz = try container.decode(Int.self, forKey: .xyz)
    self.data = try container.nestedContainer(keyedBy: dataCodingKey.self, forKey: .data)
    self.dataList = try data.decode([CustomClass].self, forKey: .dataList)
  }
}

CustomClass.swift

class CustomClass: NSObject, Decodable, MKAnnotation {

    var id: String?
    var a1: String?
    var a2: String?
    var lat: Double?
    var lon: Double?
    var coordinate: CLLocationCoordinate2D?

    enum CodingKeys: String, CodingKey {
        case id
        case a1
        case a2
        case lat
        case lon
    }

    required init(from decoder: Decoder) throws {
        let values = try decoder.container(keyedBy: CodingKeys.self)
        self.id = try values.decode(String.self, forKey: .id)
        self.a1 = try values.decodeIfPresent(String.self, forKey: .a1)
        self.a2 = try values.decodeIfPresent(String.self, forKey: .a2)
        self.lat = try values.decodeIfPresent(Double.self, forKey: .lat)
        self.lon = try values.decodeIfPresent(Double.self, forKey: .lon)
        self.coordinate = CLLocationCoordinate2D(latitude: self.lat?, longitude: self.lon?)
    }
}

我收到以下错误

非“@objc”属性“坐标”不满足“@objc”协议“MKAnnotation”的要求

我认为我遇到的问题是coordinate必须是非可选的。但是,在解码 URL1 时,latlon变量​​将始终为零,因为它们不存在于该 JSON 文件中。如何在不设置选项的情况下从这 2 个 JSON URL 解码,以便设置coordinate变量?

标签: iosswiftclasscodable

解决方案


MKAnnotation需要一个只读变量coordinate,所以使用计算属性:

var coordinate : CLLocationCoordinate2D {
    return CLLocationCoordinate2D(latitude: lat ?? 0.0, longitude: lon ?? 0.0)
}

另请参阅我对您上一个问题的回答,其中建议使用多个结构/类和泛型,如果您想避免使用选项,这是唯一的方法


推荐阅读