ทางเลือกหนึ่งคือการใช้ประเภท wrapper ที่พยายามถอดรหัสค่าที่กำหนด การจัดเก็บnil
หากไม่สำเร็จ:
struct FailableDecodable<Base : Decodable> : Decodable {
let base: Base?
init(from decoder: Decoder) throws {
let container = try decoder.singleValueContainer()
self.base = try? container.decode(Base.self)
}
}
จากนั้นเราสามารถถอดรหัสอาร์เรย์ของสิ่งเหล่านี้โดยGroceryProduct
กรอกข้อมูลในBase
ตัวยึด:
import Foundation
let json = """
[
{
"name": "Banana",
"points": 200,
"description": "A banana grown in Ecuador."
},
{
"name": "Orange"
}
]
""".data(using: .utf8)!
struct GroceryProduct : Codable {
var name: String
var points: Int
var description: String?
}
let products = try JSONDecoder()
.decode([FailableDecodable<GroceryProduct>].self, from: json)
.compactMap { $0.base } // .flatMap in Swift 4.0
print(products)
// [
// GroceryProduct(
// name: "Banana", points: 200,
// description: Optional("A banana grown in Ecuador.")
// )
// ]
จากนั้นเราจะใช้.compactMap { $0.base }
เพื่อกรองnil
องค์ประกอบ (สิ่งที่ทำให้เกิดข้อผิดพลาดในการถอดรหัส)
สิ่งนี้จะสร้างอาร์เรย์กลาง[FailableDecodable<GroceryProduct>]
ซึ่งไม่น่าจะเป็นปัญหา อย่างไรก็ตามหากคุณต้องการหลีกเลี่ยงคุณสามารถสร้างกระดาษห่อหุ้มประเภทอื่นที่ถอดรหัสและแกะแต่ละองค์ประกอบจากคอนเทนเนอร์ที่ไม่ได้ใส่กุญแจได้เสมอ:
struct FailableCodableArray<Element : Codable> : Codable {
var elements: [Element]
init(from decoder: Decoder) throws {
var container = try decoder.unkeyedContainer()
var elements = [Element]()
if let count = container.count {
elements.reserveCapacity(count)
}
while !container.isAtEnd {
if let element = try container
.decode(FailableDecodable<Element>.self).base {
elements.append(element)
}
}
self.elements = elements
}
func encode(to encoder: Encoder) throws {
var container = encoder.singleValueContainer()
try container.encode(elements)
}
}
จากนั้นคุณจะถอดรหัสเป็น:
let products = try JSONDecoder()
.decode(FailableCodableArray<GroceryProduct>.self, from: json)
.elements
print(products)
// [
// GroceryProduct(
// name: "Banana", points: 200,
// description: Optional("A banana grown in Ecuador.")
// )
// ]