我整天都在与 NWConnection 作斗争,以在长时间运行的 TCP 套接字上接收数据。由于缺乏文档,我在给自己造成以下错误后终于让它工作了:
- 数据不完整(由于只调用一次receive)
- 获取 TCP 数据无序(由于从计时器“轮询”接收......导致多个同时关闭等待获取数据)。
- 遭受无限循环(由于在接收后重新启动接收而没有检查“完整”书 - 一旦套接字从另一端终止,这是......糟糕......非常糟糕)。
我所学到的总结:
- 一旦您处于 .ready 状态,您就可以调用 receive...一次且仅一次
- 一旦收到一些数据,您可以再次调用 receive...但前提是您仍处于 .ready 状态并且 isComplete 为 false。
这是我的代码。我认为这是对的。但如果有错请告诉我:
queue = DispatchQueue(label: "hostname", attributes: .concurrent)
let serverEndpoint = NWEndpoint.Host(hostname)
guard let portEndpoint = NWEndpoint.Port(rawValue: port) else { return nil }
connection = NWConnection(host: serverEndpoint, port: portEndpoint, using: .tcp)
connection.stateUpdateHandler = { [weak self] (newState) in
switch newState {
case .ready:
debugPrint("TcpReader.ready to send")
self?.receive()
case .failed(let error):
debugPrint("TcpReader.client failed with error \(error)")
case .setup:
debugPrint("TcpReader.setup")
case .waiting(_):
debugPrint("TcpReader.waiting")
case .preparing:
debugPrint("TcpReader.preparing")
case .cancelled:
debugPrint("TcpReader.cancelled")
}
}
func receive() {
connection.receive(minimumIncompleteLength: 1, maximumLength: 8192) { (content, context, isComplete, error) in
debugPrint("\(Date()) TcpReader: got a message \(String(describing: content?.count)) bytes")
if let content = content {
self.delegate.gotData(data: content, from: self.hostname, port: self.port)
}
if self.connection.state == .ready && isComplete == false {
self.receive()
}
}
}
我认为您可以多次使用短时间连接。例如,客户端连接到主机并要求主机执行某些操作,然后告诉主机关闭连接。主机切换到等待模式以准备新连接。参见下图。
当客户端在特定时间内未向主机发送关闭连接或应答事件时,您应该使用连接计时器来关闭打开的连接。
![](https://github.com/filimo/ReaderTranslator/raw/master/files/Release_1.9_4.png)
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)