WrapperOfNSCoding 因单个值而失败

WrapperOfNSCoding fails with single value

来自

的问题
    struct WrapperOfNSCoding<Wrapped>: Codable where Wrapped: NSCoding {
        var wrapped: Wrapped

        init(_ wrapped: Wrapped) { self.wrapped = wrapped }

        init(from decoder: Decoder) throws {
            let container = try decoder.singleValueContainer()
            let data = try container.decode(Data.self)
            guard let object = NSKeyedUnarchiver.unarchiveObject(with: data) else {
                throw DecodingError.dataCorruptedError(in: container, debugDescription: "failed to unarchive an object")
            }
            guard let wrapped = object as? Wrapped else {
                throw DecodingError.typeMismatch(Wrapped.self, DecodingError.Context(codingPath: container.codingPath, debugDescription: "unarchived object type was \(type(of: object))"))
            }
            self.wrapped = wrapped
        }

        func encode(to encoder: Encoder) throws {
            let data = NSKeyedArchiver.archivedData(withRootObject: wrapped)
            var container = try encoder.singleValueContainer()
            try container.encode(data)
        }
    }

    let colors = [NSColor.red, NSColor.brown]
    print(colors)
    let w = WrapperOfNSCoding(colors[0])
    let jsonData = try! JSONEncoder().encode(w) - fails
    let jsonData = try! JSONEncoder().encode(colors.map({ WrapperOfNSCoding([=10=]) })) - succeeds

    print(jsonData)
    let colors2 = try! JSONDecoder().decode([WrapperOfNSCoding<NSColor>].self, from: jsonData).map({ [=10=].wrapped })
    print(colors2)

错误是在编码器中使用单个值时

let w = WrapperOfNSCoding(colors[0])
let jsonData = try! JSONEncoder().encode(w)

错误是

Fatal error: 'try!' expression unexpectedly raised an error: Swift.EncodingError.invalidValue(WrapperOfNSCoding #1...

这就成功了

let w = WrapperOfNSCoding([colors[0]])
let jsonData = try! JSONEncoder().encode(w)

为什么会这样

JSONEncoder 需要顶层的有效 JSON 上下文,可以是 [:](字典)或 [](数组),您可以在其中放置一个本例中的元素 string.

当您保存任何 NSCoding 对象时,编译器将该对象视为字符串。在 JSONEncoder().encode(w) 的情况下,您正在尝试编码充当 string 对象而不是常规 JSON 对象的 NSCoding 对象。

JSONEncoder().encode([w]) 的情况下,对象创建了一个数组,每个元素都有一个字符串,在您的情况下是 NSCoding

换句话说,w 数据是一个字符串,[w] 是一个数组,每个索引都有一个字符串,因此 JSONEncoder 不会给你一个错误 [w] 并给你一个 w 的错误。