使用 libev 的套接字

Posted

技术标签:

【中文标题】使用 libev 的套接字【英文标题】:sockets using libev 【发布时间】:2013-02-25 00:56:55 【问题描述】:

我正在寻找基于 libev 编写一个套接字程序。我注意到https://github.com/coolaj86/libev-examples/blob/master/src/unix-echo-server.c 中所述的几个示例使用基于 init 的回调。例如,

main() 

......
ev_io_init(&client.io, client_cb, client.fd, EV_READ|EV_WRITE);
ev_io_start(EV_A_ &server.io);



static void client_cb (EV_P_ ev_io *w, int revents)

  if (revents & EV_READ)
  
      ....
   else if (revents & EV_WRITE) 

     ......
  

我的问题来自预期的行为,例如,我在 EV_READ 中读取的所有内容都存储在链表中。假设我不断获得免费的数据包流来读取,我有机会进入 EV_WRITE 吗?我必须将我通过读取接收到的所有内容发送到另一个套接字。那么它会是一次 EV_READ 和第二次 EV_WRITE 吗?换句话说,EV_WRITE 什么时候会被解除阻塞?或者我是否需要阻止 EV_READ 才能调用 EV_WRITE。有人可以帮我理解这一点吗?

【问题讨论】:

codefundas.blogspot.com/2010/09/… 【参考方案1】:

简短地回答:如果您总是先检查一种类型的事件,然后为另一种类型设置else if,您将面临饥饿的风险。一般来说,我会检查两者,除非指定的协议无法同时激活两者。

这是一个更不确定的答案: 您问题中的链接不包含代码结构,例如您的问题。客户端https://github.com/coolaj86/libev-examples/blob/master/src/unix-echo-client.c 确实有类似的回调。您会注意到它会在写入一次后禁用写入事件。

// once the data is sent, stop notifications that
// data can be sent until there is actually more
// data to send
ev_io_stop(EV_A_ &send_w);
ev_io_set(&send_w, remote_fd, EV_READ);
ev_io_start(EV_A_ &send_w);

这看起来像是试图避免管道 READ 事件分支的饥饿。即使我对 libev 不是很熟悉,您链接到的 github 示例似乎也不是很健壮。例如static void stdin_cb (EV_P_ ev_io *w, int revents)不使用getline()的返回值来检测EOF。此外,send()recv() 套接字操作返回值不会检查读取或写入的数量(尽管在本地命名管道流上,数量很可能与请求的数量匹配)。如果稍后将其更改为基于 TCP 的连接,则检查数量至关重要。

【讨论】:

是的,我知道我用来理解的代码不是很好,但是如果您能指出我的任何其他代码,我将非常高兴。因此,从您所说的来看,如果我同时检查两者,我应该可以避免饥饿。谢谢。 是的,两者都检查。使用if() 语句而不是if-else【参考方案2】:

我认为您应该将写回调与读回调分开:

main() 
    ev_io_init(&read.io, read_cb, client.fd, EV_READ);
    ev_io_init(&write.io, writead_cb, client.fd, EV_WRITE);
    ev_io_start(EV_A_ &read.io);
    ev_io_start(EV_A_ &write.io);

这是我的解决方案。

【讨论】:

以上是关于使用 libev 的套接字的主要内容,如果未能解决你的问题,请参考以下文章

libevent 无法读取打开的套接字描述符

带有 libevent/libev 的 C 或 C++:监视 unix 套接字

在同一个套接字中使用 libevent 处理 HTTP 和 HTTPS 请求

libevent 如何检测到套接字已关闭

libevent : 如果相关套接字被本地程序关闭,是不是可以触发事件

pthread 向 libevent 添加一个套接字但退出