>来自 Realize Codable for UIColor 的问题
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($0) })) - succeeds
print(jsonData)
let colors2 = try! JSONDecoder().decode([WrapperOfNSCoding<NSColor>].self, from: jsonData).map({ $0.wrapped })
print(colors2)
错误是在编码器中使用单个值时
let w = WrapperOfNSCoding(colors[0])
let jsonData = try! JSONEncoder().encode(w)
错误是
致命错误:"尝试!"表达式意外引发错误: 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
错误。