【发布时间】:2023-03-04 18:58:01
【问题描述】:
我有这个 JSON 文件。
[
{
"name": "January",
"holidays": [
{
"name": "New Year's Day",
"date": "2019-01-01T00:00:00-0500",
"type": {
"isNationalHoliday": true,
"isRegionalHoliday": true,
"isPublicHoliday": true,
"isGovernmentHoliday": true
}
},
{
"name": "Martin Luther King Day",
"date": "2019-01-21T00:00:00-0500",
"type": {
"isNationalHoliday": true,
"isRegionalHoliday": true,
"isPublicHoliday": true,
"isGovernmentHoliday": true
}
}
]
},
{
"name": "February",
"holidays": [
{
"name": "Presidents' Day",
"date": "2019-02-18T00:00:00-0500",
"type": {
"isNationalHoliday": false,
"isRegionalHoliday": true,
"isPublicHoliday": false,
"isGovernmentHoliday": false
}
}
]
},
{
"name": "March",
"holidays": null
}
]
我创建了一个 Month 结构来解码 JSON 中的字典。
public struct Month {
public let name: String
public let holidays: [Holiday]?
}
extension Month: Decodable { }
还有一个 Year 结构来包含它们。
public struct Year {
public let months: [Month]
}
extension Year: Decodable {
public init(from decoder: Decoder) throws {
let container = try decoder.singleValueContainer()
let values = try container.decode([Month].self)
months = values
}
}
我的Holiday 结构有点复杂,因为有一个名为HolidayType 的枚举,我想在其中解码JSON 中type 字段下的值。
public struct Holiday {
public let name: String
public let date: Date
public let type: HolidayType
}
extension Holiday: Decodable { }
public enum HolidayType {
case isNationalHoliday
case isRegionalHoliday
case isPublicHoliday
case isGovernmentHoliday
enum CodingKeys: String, CodingKey {
case isNationalHoliday
case isRegionalHoliday
case isPublicHoliday
case isGovernmentHoliday
}
}
extension HolidayType: Decodable {
public init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
self = try container.decode(HolidayType.self, forKey: .isNationalHoliday)
self = try container.decode(HolidayType.self, forKey: .isRegionalHoliday)
self = try container.decode(HolidayType.self, forKey: .isPublicHoliday)
self = try container.decode(HolidayType.self, forKey: .isGovernmentHoliday)
}
}
这是我加载文件和解码的地方。
if let url = Bundle.main.url(forResource: "holidays", withExtension: "json") {
do {
let data = try Data(contentsOf: url)
let decoder = JSONDecoder()
decoder.dateDecodingStrategy = .iso8601
let year = try decoder.decode(Year.self, from: data)
print(year.months)
} catch let error {
print("Error occurred decoding JSON: \(error)")
}
} else {
print("Error occurred loading file")
}
但它失败并出现以下错误。
typeMismatch(Swift.Dictionary, Swift.DecodingError.Context(codingPath: [_JSONKey(stringValue: "索引 0", intValue: 0), CodingKeys(stringValue: "holidays", intValue: nil), _JSONKey(stringValue: "Index 0", intValue: 0), CodingKeys(stringValue: "type", intValue: nil), CodingKeys(stringValue: "isNationalHoliday", intValue: nil)], debugDescription: "预计解码 字典但找到了一个数字。”,基础错误: 无))
我不知道如何解决这个问题。我还上传了一个演示项目here。
【问题讨论】:
-
除了下面提到的枚举问题,注意顶层对象是一个数组,所以要解码
[Month].self而不是Year.self。 -
@MartinR 他有
Year的自定义解码。 -
您的
Year结构是多余的,只需解码let months = try decoder.decode([Month].self, from: data); print(months)。而仅采用Decodable的extensions 也是多余的。如果您不是结构或类的所有者,则仅采用协议的空扩展才有意义 -
@Sulthan:你说得对,我没注意到。
-
@Sulthan 恕我直言,如果有另一个属性
name或number但不要在层次结构的顶层包装单个属性是有意义的。
标签: ios json swift boolean jsonencoder