希望使用JSONEncoder+Encodable将对象编码为自定义结构。
struct Foo: Encodable {
var name: String?
var bars: [Bar]?
}
struct Bar: Encodable {
var name: String?
var value: String?
}
let bar1 = Bar(name: "bar1", value: "barvalue1")
let bar2 = Bar(name: "bar2", value: "barvalue2")
let foo = Foo(name: "foovalue", bars: [bar1, bar2])foo的默认编码方法提供:
let encoder = JSONEncoder()
encoder.outputFormatting = .prettyPrinted
let data = try encoder.encode(foo)
print(String(data: data, encoding: .utf8)!)输出:
{
"name": "foovalue",
"bars": [
{
"name": "bar1",
"value": "barvalue1"
},
{
"name": "bar2",
"value": "barvalue2"
}
]
}在自定义输出中,我希望使用属性name的值作为键,使用rest值作为上述键的值。这同样适用于嵌套对象。所以我希望输出是:
{
"foovalue": [
{
"bar1": "barvalue1"
},
{
"bar2": "barvalue2"
}
]
}问题是Encodable/JSONEncoder是否支持这一点。现在,我只需要处理第一个输出字典,并通过迭代键来重新构造它。
发布于 2018-02-10 08:09:23
如果您希望保留Foo和Bar Encodable,则可以通过提供自定义encode(to:)来实现这一点,该encode(to:)使用值为name的特定编码键
private struct StringKey: CodingKey {
let stringValue: String
var intValue: Int? { return nil }
init(_ string: String) { stringValue = string }
init?(stringValue: String) { self.init(stringValue) }
init?(intValue: Int) { return nil }
}
struct Foo: Encodable {
var name: String
var bars: [Bar]
func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: StringKey.self)
try container.encode(bars, forKey: StringKey(name))
}
}
struct Bar : Encodable {
var name: String
var value: String
func encode(to encoder: Encoder) throws {
var container = encoder.container(keyedBy: StringKey.self)
try container.encode(value, forKey: StringKey(name))
}
}StringKey可以接受任何String值,允许根据需要任意编码。
https://stackoverflow.com/questions/48718091
复制相似问题