ios - 使用 Codable 从 2 个不同的 JSON 文件实例化单个类,而不使用选项

标签 ios swift class codable

我正在使用提供 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 URL 来使用嵌套 dataList 列表中的项目创建单个 Codable CustomClass 对象,因此我创建了一个 Feed 结构来处理这 2 个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?)
    }
}

我收到以下错误

Non-'@objc' property 'coordinate' does not satisfy requirement of '@objc' protocol 'MKAnnotation'

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

最佳答案

MKAnnotation 需要一个只读变量坐标,因此请使用计算属性:

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

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

关于ios - 使用 Codable 从 2 个不同的 JSON 文件实例化单个类,而不使用选项,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52346770/

相关文章:

objective-c - 在 Objective-C 中使用 Swift 中的 JSQSystemSoundPlayer 时,我没有看到可见的 @interface

ios - 如何使用 Swift 从 Objective-C 启动实例类型

c# - 初学者 : Extending a class in C#, 我做错了吗?

php - 我如何在 PHP 中组织类并从类中访问数据库

ios UILabel 通过 UILabel 大小变化实现动态字体大小变化

ios - 关于在 Swift 中触摸一次屏幕时识别手势的问题

swift - 根据 Swift 中的位置调用电话

ios - 为 iPhone 5 加载不同的 xib 失败

ios - Xcode-sqlite3.dylib : illegal multi-thread access to database connection

c++ - 一个指针,c++中的两个不同的类