在 Swift 中接收 Websocket 数据

问题描述

我将在 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 结构不符合 CustomStringConvertibleCustomDebugStringConvertibleTextOutputStreamable,所以 "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 的实例。

没有。 WebSocketControllerPerson 结构的完全限定名称的一部分。您的解码数组中只有一个实例,它是 Person 的实例,正如您所料!

我如何访问这些数据?

访问其名称:

if let firstPerson = people.first {
    let firstPersonsName = firstPerson.name
}

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...