这是一篇很好的博客文章,其中包含您问题的答案:source https://benscheirman.com/2017/06/swift-json/
向下滚动到继承,您将看到以下内容:
假设我们有以下类:
class Person : Codable {
var name: String?
}
class Employee : Person {
var employeeID: String?
}
我们通过继承 Person 类来获得 Codable 一致性,但是如果我们尝试对 Employee 实例进行编码会发生什么?
let employee = Employee()
employee.employeeID = "emp123"
employee.name = "Joe"
let encoder = JSONEncoder()
encoder.outputFormatting = .prettyPrinted
let data = try! encoder.encode(employee)
print(String(data: data, encoding: .utf8)!)
{
"name" : "Joe"
}
这不是预期的结果,因此我们必须添加一个自定义实现,如下所示:
class Person : Codable {
var name: String?
private enum CodingKeys : String, CodingKey {
case name
}
func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: CodingKeys.self)
try container.encode(name, forKey: .name)
}
}
子类的故事相同:
class Employee : Person {
var employeeID: String?
private enum CodingKeys : String, CodingKey {
case employeeID = "emp_id"
}
override func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: CodingKeys.self)
try container.encode(employeeID, forKey: .employeeID)
}
}
结果将是:
{
"emp_id" : "emp123"
}
这又不是预期的结果,所以这里我们使用遗产通过致电super.
// Employee.swift
override func encode(to encoder: Encoder) throws {
try super.encode(to: encoder)
var container = encoder.container(keyedBy: CodingKeys.self)
try container.encode(employeeID, forKey: .employeeID)
}
这最终给了我们从一开始就真正想要的东西:
{
"name": "Joe",
"emp_id": "emp123"
}
如果您对扁平化结果不满意,还有一些关于如何避免这种情况的提示。
所有的功劳都归功于写这篇博文的人,并表示我的感谢。
希望对你也有帮助,加油!