17

我正在使用 Swift 4 和 Xcode 9.2 编写程序。我在编写可编码类(确切地说是类,而不是结构)时遇到了困难。当我试图从另一个类继承一个类时,JSONEncoder 不会从子类(子类)中获取所有属性。请看这个:

class BasicData: Encodable {

    let a: String
    let b: String

    init() {
        a = "a"
        b = "b"
    }
}

class AdditionalData: BasicData {

    let c: String

    init(c: String) {
        self.c = c
    }

}

let encode = AdditionalData(c: "c")

do {
    let data = try JSONEncoder().encode(encode)
    let string = String(data: data, encoding: .utf8)
    if let string = string {
        print(string)
    }
} catch {
}

它会打印这个:{"a":"a","b":"b"}

但我需要这个:{"a":"a","b":"b","c":"c"}

看起来c阶级的财产AdditionalData只是不知何故丢失了。

所以问题是:如果我有使用协议 Encodable 签名的类,如何正确地制作子类(此类的子类,继承)类?

我将感谢任何帮助或建议。

4

2 回答 2

29

EncodableDecodable涉及一些代码合成,其中编译器基本上为您编写代码。当您遵守BasicDataEncodable,这些方法将写入BasicData类,因此它们不知道子类定义的任何附加属性。您必须覆盖encode(to:)子类中的方法:

class AdditionalData: BasicData {
    let c: String
    let d: Int

    init(c: String, d: Int) {
        self.c = c
        self.d = d
    }

    private enum CodingKeys: String, CodingKey {
        case c, d
    }

    override func encode(to encoder: Encoder) throws {
        try super.encode(to: encoder)
        var container = encoder.container(keyedBy: CodingKeys.self)
        try container.encode(self.c, forKey: .c)
        try container.encode(self.d, forKey: .d)
    }
}

请参阅此问题以了解与Decodable.

于 2018-01-14T23:46:44.673 回答
0

就我而言,基类实际上不需要是 Codable 。只有子类需要是可编码的。在这种情况下,不要将基类声明为 Codable,而只声明子类。通过这样做,您不需要执行任何这些 encode/codingKeys/init(from: Decoder) 样板文件。希望这对某些人有所帮助。

于 2020-07-23T19:25:10.867 回答