问题描述
我将在 from this question 进行此操作,因为焦点已改变。
我正在尝试通过 websocket 从蒸汽服务器发送字符串数据。客户端是主要问题所在。此代码成功接收字符串,该字符串应为 JSON(但不能绝对保证 - 超出范围)。
switch message {
case .data(let data):
print("data: \(data)")
case .string(let str):
// let data = str.message(using: .utf8)
let jsonData = Data(str.utf8)
print("string: \(jsonData)")
do {
struct Person : Codable {
var name: String
}
let decoder = JSONDecoder()
let people = try decoder.decode([Person].self,from: jsonData)
print("result: \(people)")
} catch {
print(error.localizedDescription)
}
}
经过一些非常有用的指导后,发送诸如 "{\"name\": \"Bobberoo\"}"
之类的字符串将打印出来
string: 20 bytes
The data Couldn’t be read because it isn’t in the correct format.
如果我用大括号 "[{\"name\": \"Bobberoo\"}]"
将它包装起来会产生更有用但仍然令人困惑(对我来说)的输出:
result: [wb2_socket_client.WebSocketController.(unkNown context at $101a35028).(unkNown context at $101a350c0).(unkNown context at $101a35158).Person(name: "Bobberoo")]
显然,解码正在发生,但它包含在这些上下文中。这些是什么?我可以看到第一个是 WebSocketController 的实例。我如何访问这些数据。
作为一个非煽动性的旁白:管理 JSON 在任何数量的上下文中都是一项微不足道的操作。 Python/Flask、Node、Ruby/Rails 等等;我已经使用了所有这些并且实现这种交互是微不足道的。在 Swift 中,这是一个可怕的、记录不足的噩梦。至少,这是我的经验。为什么?我知道该语言是类型安全的,但这太荒谬了。
解决方法
error.localizedDescription
不会向您提供对调试有用的错误消息。另一方面,如果您直接打印 error
:
print(error)
你会得到类似“预期解码数组但找到字典”的内容,这正是
{
"name": "Bobberoo"
}
您正在解码 [Person].self
,即 Person
的 数组,但您的 JSON 根不是 JSON 数组。如果你这样做了,上面的 JSON 可以被解码:
let people = try decoder.decode(Person.self,from: jsonData)
显然,解码正在发生,但它包含在这些上下文中。它们是什么?
这是一个类型的默认字符串表示。您的 Person
结构不符合 CustomStringConvertible
或 CustomDebugStringConvertible
或 TextOutputStreamable
,所以 "an unspecified result is supplied automatically by the Swift standard library"(链接指向 String.init(reflecting:)
,大概会被调用当您print
Person
的数组) 并用作字符串表示时的某个地方。
据我所知,它当前的实现是结构的完全限定名称 - 从模块开始,然后是顶级类,然后是每个封闭范围,以结构名称结尾,然后是结构的成员括号。事实证明,封闭的作用域没有“名称”,因此仅称为 (unknown context at xxxxx)
。这都是非常多的实现细节,以及您不应该关心的事情。
您应该做的是提供 CustomStringConvertible
的实现:
struct Person: CustomStringConvertible {
...
var description: String { "name: \(name)" }
}
现在打印 people
给出:
[name: Bobberoo]
我可以看到第一个是 WebSocketController
的实例。
没有。 WebSocketController
是 Person
结构的完全限定名称的一部分。您的解码数组中只有一个实例,它是 Person
的实例,正如您所料!
我如何访问这些数据?
访问其名称:
if let firstPerson = people.first {
let firstPersonsName = firstPerson.name
}