Swift Codable:如何将顶级数据编码到嵌套容器中
Swift Codable: How to encode top-level data into nested container
我的应用程序使用的服务器 returns JSON 看起来像这样:
{
"result":"OK",
"data":{
// Common to all URLs
"user": {
"name":"John Smith" // ETC...
},
// Different for each URL
"data_for_this_url":0
}
}
如您所见,URL 特定信息与通用 user
词典存在于同一个词典中。
目标:
- 将此 JSON 解码为 classes/structs。
- 因为
user
很常见,所以我希望它位于顶层 class/struct。
- 编码为新格式(例如 plist)。
- 我需要保留原来的结构。 (即从顶级
user
信息和子对象的信息重新创建 data
字典)
问题:
重新编码数据时,我无法同时将 user
字典(来自顶级对象)和 URL 特定数据(来自子对象)写入编码器。
要么user
覆盖其他数据,要么其他数据覆盖user
。我不知道如何组合它们。
这是我目前的情况:
// MARK: - Common User
struct User: Codable {
var name: String?
}
// MARK: - Abstract Response
struct ApiResponse<DataType: Codable>: Codable {
// MARK: Properties
var result: String
var user: User?
var data: DataType?
// MARK: Coding Keys
enum CodingKeys: String, CodingKey {
case result, data
}
enum DataDictKeys: String, CodingKey {
case user
}
// MARK: Decodable
init(from decoder: Decoder) throws {
let baseContainer = try decoder.container(keyedBy: CodingKeys.self)
self.result = try baseContainer.decode(String.self, forKey: .result)
self.data = try baseContainer.decodeIfPresent(DataType.self, forKey: .data)
let dataContainer = try baseContainer.nestedContainer(keyedBy: DataDictKeys.self, forKey: .data)
self.user = try dataContainer.decodeIfPresent(User.self, forKey: .user)
}
// MARK: Encodable
func encode(to encoder: Encoder) throws {
var baseContainer = encoder.container(keyedBy: CodingKeys.self)
try baseContainer.encode(self.result, forKey: .result)
// MARK: - PROBLEM!!
// This is overwritten
try baseContainer.encodeIfPresent(self.data, forKey: .data)
// This overwrites the previous statement
var dataContainer = baseContainer.nestedContainer(keyedBy: DataDictKeys.self, forKey: .data)
try dataContainer.encodeIfPresent(self.user, forKey: .user)
}
}
示例:
在下面的例子中,重新编码的plist不包含order_count
,因为它被包含user
.
的字典覆盖了
// MARK: - Concrete Response
typealias OrderDataResponse = ApiResponse<OrderData>
struct OrderData: Codable {
var orderCount: Int = 0
enum CodingKeys: String, CodingKey {
case orderCount = "order_count"
}
}
let orderDataResponseJson = """
{
"result":"OK",
"data":{
"user":{
"name":"John"
},
"order_count":10
}
}
"""
// MARK: - Decode from JSON
let jsonData = orderDataResponseJson.data(using: .utf8)!
let response = try JSONDecoder().decode(OrderDataResponse.self, from: jsonData)
// MARK: - Encode to PropertyList
let plistEncoder = PropertyListEncoder()
plistEncoder.outputFormat = .xml
let plistData = try plistEncoder.encode(response)
let plistString = String(data: plistData, encoding: .utf8)!
print(plistString)
// 'order_count' is not included in 'data'!
/*
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>data</key>
<dict>
<key>user</key>
<dict>
<key>name</key>
<string>John</string>
</dict>
</dict>
<key>result</key>
<string>OK</string>
</dict>
</plist>
*/
我刚刚在浏览编码器协议时顿悟了。
KeyedEncodingContainerProtocol.superEncoder(forKey:)
方法正是针对这种情况。
此方法 returns 一个单独的 Encoder
可以收集多个项目 and/or 嵌套容器,然后将它们编码成一个键。
对于这种特定情况,顶级 user
数据可以通过简单地调用它自己的 encode(to:)
方法和新的 superEncoder
方法来编码。然后,也可以使用编码器创建嵌套容器,以正常使用。
问题解答
// MARK: - Encodable
func encode(to encoder: Encoder) throws {
var baseContainer = encoder.container(keyedBy: CodingKeys.self)
try baseContainer.encode(self.result, forKey: .result)
// MARK: - PROBLEM!!
// // This is overwritten
// try baseContainer.encodeIfPresent(self.data, forKey: .data)
//
// // This overwrites the previous statement
// var dataContainer = baseContainer.nestedContainer(keyedBy: DataDictKeys.self, forKey: .data)
// try dataContainer.encodeIfPresent(self.user, forKey: .user)
// MARK: - Solution
// Create a new Encoder instance to combine data from separate sources.
let dataEncoder = baseContainer.superEncoder(forKey: .data)
// Use the Encoder directly:
try self.data?.encode(to: dataEncoder)
// Create containers for manually encoding, as usual:
var userContainer = dataEncoder.container(keyedBy: DataDictKeys.self)
try userContainer.encodeIfPresent(self.user, forKey: .user)
}
输出:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>data</key>
<dict>
<key>order_count</key>
<integer>10</integer>
<key>user</key>
<dict>
<key>name</key>
<string>John</string>
</dict>
</dict>
<key>result</key>
<string>OK</string>
</dict>
</plist>
很好的问题和解决方案,但如果您想简化它,您可以使用我写的 KeyedCodable。 Codable 的整个实现看起来像这样(当然,OrderData 和 User 保持不变):
struct ApiResponse<DataType: Codable>: Codable {
// MARK: Properties
var result: String!
var user: User?
var data: DataType?
enum CodingKeys: String, KeyedKey {
case result
case user = "data.user"
case data
}
}
我的应用程序使用的服务器 returns JSON 看起来像这样:
{
"result":"OK",
"data":{
// Common to all URLs
"user": {
"name":"John Smith" // ETC...
},
// Different for each URL
"data_for_this_url":0
}
}
如您所见,URL 特定信息与通用 user
词典存在于同一个词典中。
目标:
- 将此 JSON 解码为 classes/structs。
- 因为
user
很常见,所以我希望它位于顶层 class/struct。
- 因为
- 编码为新格式(例如 plist)。
- 我需要保留原来的结构。 (即从顶级
user
信息和子对象的信息重新创建data
字典)
- 我需要保留原来的结构。 (即从顶级
问题:
重新编码数据时,我无法同时将 user
字典(来自顶级对象)和 URL 特定数据(来自子对象)写入编码器。
要么user
覆盖其他数据,要么其他数据覆盖user
。我不知道如何组合它们。
这是我目前的情况:
// MARK: - Common User
struct User: Codable {
var name: String?
}
// MARK: - Abstract Response
struct ApiResponse<DataType: Codable>: Codable {
// MARK: Properties
var result: String
var user: User?
var data: DataType?
// MARK: Coding Keys
enum CodingKeys: String, CodingKey {
case result, data
}
enum DataDictKeys: String, CodingKey {
case user
}
// MARK: Decodable
init(from decoder: Decoder) throws {
let baseContainer = try decoder.container(keyedBy: CodingKeys.self)
self.result = try baseContainer.decode(String.self, forKey: .result)
self.data = try baseContainer.decodeIfPresent(DataType.self, forKey: .data)
let dataContainer = try baseContainer.nestedContainer(keyedBy: DataDictKeys.self, forKey: .data)
self.user = try dataContainer.decodeIfPresent(User.self, forKey: .user)
}
// MARK: Encodable
func encode(to encoder: Encoder) throws {
var baseContainer = encoder.container(keyedBy: CodingKeys.self)
try baseContainer.encode(self.result, forKey: .result)
// MARK: - PROBLEM!!
// This is overwritten
try baseContainer.encodeIfPresent(self.data, forKey: .data)
// This overwrites the previous statement
var dataContainer = baseContainer.nestedContainer(keyedBy: DataDictKeys.self, forKey: .data)
try dataContainer.encodeIfPresent(self.user, forKey: .user)
}
}
示例:
在下面的例子中,重新编码的plist不包含order_count
,因为它被包含user
.
// MARK: - Concrete Response
typealias OrderDataResponse = ApiResponse<OrderData>
struct OrderData: Codable {
var orderCount: Int = 0
enum CodingKeys: String, CodingKey {
case orderCount = "order_count"
}
}
let orderDataResponseJson = """
{
"result":"OK",
"data":{
"user":{
"name":"John"
},
"order_count":10
}
}
"""
// MARK: - Decode from JSON
let jsonData = orderDataResponseJson.data(using: .utf8)!
let response = try JSONDecoder().decode(OrderDataResponse.self, from: jsonData)
// MARK: - Encode to PropertyList
let plistEncoder = PropertyListEncoder()
plistEncoder.outputFormat = .xml
let plistData = try plistEncoder.encode(response)
let plistString = String(data: plistData, encoding: .utf8)!
print(plistString)
// 'order_count' is not included in 'data'!
/*
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>data</key>
<dict>
<key>user</key>
<dict>
<key>name</key>
<string>John</string>
</dict>
</dict>
<key>result</key>
<string>OK</string>
</dict>
</plist>
*/
我刚刚在浏览编码器协议时顿悟了。
KeyedEncodingContainerProtocol.superEncoder(forKey:)
方法正是针对这种情况。
此方法 returns 一个单独的 Encoder
可以收集多个项目 and/or 嵌套容器,然后将它们编码成一个键。
对于这种特定情况,顶级 user
数据可以通过简单地调用它自己的 encode(to:)
方法和新的 superEncoder
方法来编码。然后,也可以使用编码器创建嵌套容器,以正常使用。
问题解答
// MARK: - Encodable
func encode(to encoder: Encoder) throws {
var baseContainer = encoder.container(keyedBy: CodingKeys.self)
try baseContainer.encode(self.result, forKey: .result)
// MARK: - PROBLEM!!
// // This is overwritten
// try baseContainer.encodeIfPresent(self.data, forKey: .data)
//
// // This overwrites the previous statement
// var dataContainer = baseContainer.nestedContainer(keyedBy: DataDictKeys.self, forKey: .data)
// try dataContainer.encodeIfPresent(self.user, forKey: .user)
// MARK: - Solution
// Create a new Encoder instance to combine data from separate sources.
let dataEncoder = baseContainer.superEncoder(forKey: .data)
// Use the Encoder directly:
try self.data?.encode(to: dataEncoder)
// Create containers for manually encoding, as usual:
var userContainer = dataEncoder.container(keyedBy: DataDictKeys.self)
try userContainer.encodeIfPresent(self.user, forKey: .user)
}
输出:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>data</key>
<dict>
<key>order_count</key>
<integer>10</integer>
<key>user</key>
<dict>
<key>name</key>
<string>John</string>
</dict>
</dict>
<key>result</key>
<string>OK</string>
</dict>
</plist>
很好的问题和解决方案,但如果您想简化它,您可以使用我写的 KeyedCodable。 Codable 的整个实现看起来像这样(当然,OrderData 和 User 保持不变):
struct ApiResponse<DataType: Codable>: Codable {
// MARK: Properties
var result: String!
var user: User?
var data: DataType?
enum CodingKeys: String, KeyedKey {
case result
case user = "data.user"
case data
}
}