我想在 Swift中表示一个通用的JSON对象: let foo: [String: Any] = [ "foo": 1, "bar": "baz",] 但编译器建议的[String:Any]类型并不能很好地工作.我无法检查该类型的两个实例是否相等,例如,虽然这应
let foo: [String: Any] = [
"foo": 1,
"bar": "baz",
]
但编译器建议的[String:Any]类型并不能很好地工作.我无法检查该类型的两个实例是否相等,例如,虽然这应该可以使用两个JSON树.
什么也行不通的是使用Codable机制将该值编码为JSON字符串:
let encoded = try JSONEncoder().encode(foo)
哪个因错误而爆炸:
fatal error: Dictionary<String, Any> does not conform to Encodable because Any does not conform to Encodable.
我知道我可以引入一个精确的类型,但我遵循通用的JSON结构.我甚至尝试为泛型JSON引入一种特定类型:
enum JSON {
case string(String)
case number(Float)
case object([String:JSON])
case array([JSON])
case bool(Bool)
case null
}
但是当为这个枚举实现Codable时,我不知道如何实现encode(to :),因为一个带键的容器(用于编码对象)需要一个特定的CodingKey参数,我不知道如何获得它.
是否真的不可能创建一个Equatable通用JSON树并使用Codable对其进行编码?
我们将使用通用字符串作为编码键:extension String: CodingKey {
public init?(stringValue: String) {
self = stringValue
}
public var stringValue: String {
return self
}
public init?(intValue: Int) {
return nil
}
public var intValue: Int? {
return nil
}
}
剩下的只是获取正确类型的容器并将值写入其中.
extension JSON: Encodable {
public func encode(to encoder: Encoder) throws {
switch self {
case .string(let string):
var container = encoder.singleValueContainer()
try container.encode(string)
case .number(let number):
var container = encoder.singleValueContainer()
try container.encode(number)
case .object(let object):
var container = encoder.container(keyedBy: String.self)
for (key, value) in object {
try container.encode(value, forKey: key)
}
case .array(let array):
var container = encoder.unkeyedContainer()
for value in array {
try container.encode(value)
}
case .bool(let bool):
var container = encoder.singleValueContainer()
try container.encode(bool)
case .null:
var container = encoder.singleValueContainer()
try container.encodeNil()
}
}
}
鉴于此,我相信你可以自己实现Decodable和Equatable.
请注意,如果您尝试将除数组或对象之外的任何内容编码为顶级元素,则会崩溃.
