- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我们正在用 Haskell 编写一个消息代理 (HMB) 。因此,从套接字( Data.Binary )收到消息后,必须对其进行解析( Network.Socket )。到目前为止,我们一直在环回(localhost)上进行测试 - 用于生成和解析消息。这很安静。如果我们通过从另一台机器生成消息来进行基准测试,我们将面临问题:解析器突然没有足够的字节来解析。
每条消息的前 4 个字节定义了消息的长度,从而描述了要解析的消息。如上所述,我们使用 Data.Binary 进行解析 - 所以这是懒惰的。出于测试目的,我们使用 Cereal 库将前 4 个字节的解析切换为严格。这同样是一个问题。我们现在甚至尝试仅使用 Cereal 完全解析请求,但问题仍然存在。
在代码中您会看到我们进行了线程处理。但是,我们也尝试过不使用 channel (单线程设置),但这也没有解决问题。
这是代码的一部分(线程1),其中接收到的字节被写入 channel 以供进一步使用/解析。 (如前所述,如果我们省略 channel 并直接解析输入,则不会发生任何变化):
runConnection :: (Socket, SockAddr) -> RequestChan -> Bool -> IO()
runConnection conn chan False = return ()
runConnection conn chan True = do
r <- recvFromSock conn
case (r) of
Left e -> do
handleSocketError conn e
runConnection conn chan False
Right input -> do
threadDelay 5000 -- THIS FIXES THE PROBLEM!?
writeToReqChan conn chan input
runConnection conn chan True
这是解析输入的部分(Thread2):
runApiHandler :: RequestChan -> ResponseChan -> IO()
runApiHandler rqChan rsChan = do
(conn, req) <- readChan rqChan
case readRequest req of -- readRequest IS THE PARSER
Left (bs, bo, e) -> handleHandlerError conn $ ParseRequestError e
Right (bs, bo, rm) -> do
res <- handleRequest rm
case res of
Left e -> handleHandlerError conn e
Right bs -> writeToResChan conn rsChan bs
runApiHandler rqChan rsChan
现在我发现,如果解析过程延迟一点(请参阅第一个代码块中的 threadDelay),一切都会正常。这基本上意味着,解析器不会等待从套接字接收的字节。
这是为什么呢?为什么解析器不等待套接字有足够的字节?我们的设置是否存在一般性错误?
最佳答案
我敢打赌,这个问题与解析器无关,而是由于 UNIX 套接字的阻塞语义造成的。
当一个环回时接口(interface)可能会将数据包直接从发送者传递到接收者,以太网接口(interface)可能需要分解数据包以适应最大链路的传输单元 (MTU)。这称为数据包碎片。
长度
recv
的参数系统调用仅仅是接收长度的上限(例如目标缓冲区的大小);这调用产生的数据可能比您要求的要少。引用联机帮助页,
If no messages are available at the socket, the receive calls wait for a message to arrive, unless the socket is nonblocking (see fcntl(2)), in which case the value -1 is returned and the external variable errno is set to EAGAIN or EWOULDBLOCK. The receive calls normally return any data available, up to the requested amount, rather than waiting for receipt of the full amount requested.
因此,您可能需要多次 recv
调用来检索整个数据包。如果您延迟 recv
,则您的示例将有效,因为操作系统可以重新组装原始数据包,因为所有片段在请求时均已到达。
如meiersi指出,Haskell 世界中开发了多种流 I/O 库来解决这个问题等。其中包括pipes
, conduit
, io-streams
, 和别的。根据您的目标,这可能是处理此问题的自然方法。
关于sockets - 从 Socket 解析 ByteString 失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30375962/
基于 socket.io 的官方网站 http://socket.io/#how-to-use , 我找不到任何术语。socket.emit 、 socket.on 和 socket.send 之间有
我正在使用 lua-socket 3.0rc1.3(Ubuntu Trusty 附带)和 lua 5.1。我正在尝试监听 unix 域套接字,我能找到的唯一示例代码是 this -- send std
这两者有什么区别? 我注意到如果我在一个工作程序中从 socket.emit 更改为 socket.send ,服务器无法接收到消息,虽然我不明白为什么。 我还注意到,在我的程序中,如果我从 sock
使用套接字在两台服务器之间发送数据是个好主意,还是应该使用 MQ 之类的东西来移动数据。 我的问题:套接字是否可靠,如果我只需要一次/有保证的数据传输? 还有其他解决方案吗? 谢谢。 最佳答案 套接字
引自 this socket tutorial : Sockets come in two primary flavors. An active socket is connected to a
我已经安装了在端口81上运行的流服务器“Lighttpd”(light-tpd)。 我有一个C程序,它使用套接字api创建的服务器套接字在端口80上监听http请求。 我希望从客户端收到端口80上的请
这是我正在尝试做的事情: 当有新消息可用时,服务器会将消息发送给已连接的客户端。另一方面,客户端在连接时尝试使用send()向服务器发送消息,然后使用recv()接收消息,此后,客户端调用close(
如何将消息发送到动态 session 室,以及当服务器收到该消息时,如何将该消息发送到其他成员所在的同一个 session 室? table_id是房间,它将动态设置。 客户: var table_i
这是我尝试但不起作用的方法。我可以将传入的消息从WebSocket连接转发到NetSocket,但是只有NetSocket收到的第一个消息才到达WebSocket后面的客户端。 const WebSo
我正在实现使用boost将xml发送到客户端的服务器。我面临的问题是缓冲区不会立即发送并累积到一个点,然后发送整个内容。这在我的客户端造成了一个问题,当它解析xml时,它可能具有不完整的xml标记(不
尝试使用Nginx代理Gunicorn套接字。 /etc/systemd/system/gunicorn.service文件 [Unit] Description=gunicorn daemon Af
我正在使用Lua套接字和TCP制作像聊天客户端和服务器这样的IRC。我要弄清楚的主要事情是如何使客户端和服务器监听消息并同时发送它们。由于在服务器上执行socket:accept()时,它将暂停程序,
我看了一下ZMQ PUSH/PULL套接字,尽管我非常喜欢简单性(特别是与我现在正在通过UDP套接字在系统中实现的自定义碎片/ack相比),但我还是希望有自定义负载平衡功能,而不是幼稚的回合-robi
我正在编写一个应用程序,其中有多个 socket.io 自定义事件,并且所有工作正常,除了这个: socket.on("incomingImg", function(data) {
在我的应用程序中,我向服务器发送了两条小消息(类似 memcached 的服务)。在类似 Python 的伪代码中,这看起来像: sock.send("add some-key 0") ignored
很抱歉再次发布此问题,但大多数相关帖子都没有回答我的问题。我在使用 socket.io 的多个连接时遇到问题我没有使用“socket.socket.connect”方法,但我从第一次连接中得到了反馈。
我尝试使用 socket.io 客户端连接到非 socket.io websocket 服务器。但我做不到。我正在尝试像这样连接到套接字服务器: var socket = io.connect('ws
我遇到了一个奇怪的问题。在我非常基本的服务器中,我有: server.listen(8001); io.listen(server); var sockets = io.sockets; 不幸的是,套
我正在使用带套接字 io 的sailsjs。帆的版本是 0.10.5。我有以下套接字客户端进行测试: var socketIOClient = require('socket.io-client');
这个问题在这里已经有了答案: What is the fundamental difference between WebSockets and pure TCP? (4 个答案) 关闭 4 年前。
我是一名优秀的程序员,十分优秀!