gpt4 book ai didi

json - decoder.container(keyedBy :) throws DecodingError. typeMismatch 错误。编码错误?

转载 作者:搜寻专家 更新时间:2023-10-31 22:25:58 25 4
gpt4 key购买 nike

我正在使用 Codable 协议(protocol)从 Web API 解码 JSON。我的此 API 的 Swift 数据模型包括类继承(子类)和组合(对象作为其他对象的属性)。在 JSON 中,相同的属性名称可能表示一个完整的对象,或者表示该对象在数据库中的 ID 的单个字符串。

据我所知,使用 Codable 处理此类 JSON 的唯一模式是在对象的初始化器 init(from decoder: Decoder) 中“手动”进行解码>,并首先尝试解码整个对象。如果失败(通过抛出必须捕获的错误),则重试解码与 String 相同的属性。

只要包含 varient 属性的对象不是另一个 Decodable 类的子类,这种方法就可以正常工作。在这种情况下,解码基类的属性将在调用函数 decoder.container(keyedBy:) 时抛出错误 DecodingError.typeMismatch

请参阅下面的示例代码。

这是一个已知错误吗?和/或在这种情况下我是否缺少另一种解码方法?

顺便说一句,如果在 DecodingError.typeMismatch 错误被抛出之后调用 decoder.container(keyedBy:),那么在单个函数中也会抛出同样的错误,即使如果该错误被捕获。

import Foundation

// A `Base` class
class Base: Codable {
var baseProperty: String? = "baseProperty"
init() {}

private enum CodingKeys: String, CodingKey { case baseProperty }

required init(from decoder: Decoder) throws {
//===>> The next line will throw DecodingError.typeMismatch
let container = try decoder.container(keyedBy: CodingKeys.self)
baseProperty = try container.decode(String.self, forKey: .baseProperty)
}
}

// A Subclass of `Base`
class Sub: Base {
// An `Other` class which is a property of the `Sub` class
class Other: Codable { var id: String? = "otherID" }
var subProperty: Other? = nil
override init() { super.init() }

private enum CodingKeys: String, CodingKey { case subProperty }

required init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
do { subProperty = try container.decode(Other.self,
forKey: .subProperty)
}
catch { // We didn't find a whole `Other` object in the JSON; look for a `String` instead
let s = try container.decode(String.self, forKey: .subProperty)
subProperty = Other()
subProperty?.id = s
}
try super.init(from: decoder)
}
}

// Some sample JSON data:
let json = """
{"baseProperty" : "baseProperty",
"subProperty" : "someIDString"}
""".data(using: .utf8)!

// MAIN program -----------------------------------------------------
// Decode the JSON to produce a new Sub class instance
do {
_ = try JSONDecoder().decode(Sub.self, from: json)
}
catch DecodingError.typeMismatch( _, let context) {
print("DecodingError.typeMismatch: \(context.debugDescription)")
print("DecodingError.Context: codingPath:")
for i in 0..<context.codingPath.count { print(" [\(i)] = \(context.codingPath[i])") }
}

最佳答案

is a known bug ,已在 this pull request 中修复(并将使其进入 Swift 4.1)。

问题基本上是当 Other 的解码失败时,解码器将忘记从其内部堆栈中弹出其容器,因此意味着任何 future 的解码开始.subProperty 的嵌套容器中;因此,为什么在尝试从那里解码对象时会出现类型不匹配错误(因为只有一个字符串!)。

在修复之前,一种解决方法是不使用decode(_:forKey:);得到一个 super 解码器,然后尝试从中解码一个Other

所以替换这个:

subProperty = try container.decode(Other.self, forKey: .subProperty)

用这个:

let subPropertyDecoder = try container.superDecoder(forKey: .subProperty)
subProperty = try Other(from: subPropertyDecoder)

这是可行的,因为现在我们有了一个全新的解码器实例,我们不能破坏主解码器的堆栈。

关于json - decoder.container(keyedBy :) throws DecodingError. typeMismatch 错误。编码错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47518209/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com