c - 在套接字上使用 O_NONBLOCK 后,有没有办法避免 HUP?

标签 c linux sockets nonblocking

当我在阻塞模式下使用套接字时,我可以有一个简单的系统来做这样的事情:

client            server

A -------------------> B
      register

A <------------------> B
   (various messages)

A -------------------> B
      unregister

unregister 消息发送后,进程 A 可以立即退出,但 B 仍按预期收到消息。

如果我在 A 的套接字上打开非阻塞模式,B 永远不会收到 unregister 如果 A 发送该消息然后立即退出(我通过添加 sleep(1) 在发送 unregister 之后,在这种情况下它会按预期工作。)因此,或多或少,我的客户端无法完全取消注册。

注意:当 B poll() A 的套接字时,我得到一个挂断信号 (POLLHUP) 而不是最后一个 unregister 消息,然后挂断。

我试图添加一个调用来重新打开阻塞模式,但不知何故它没有任何区别。我使用以下代码来更改阻塞模式:

int optval(0 or 1);
ioctl(get_socket(), FIONBIO, &optval);

以防万一,我也尝试过使用 fcntl(),尽管我确信就内核而言调整了相同的标志。

int flags(fcntl(get_socket(), F_GETFL));
flags |= O_NONBLOCK;   // use this line to turn ON
flags &= ~O_NONBLOCK;  // use this line to turn OFF
fcntl(get_socket(), F_SETFL, flags);

附带说明一下,我使用 read()write() 函数发送和接收消息。


更新:

对于那些感兴趣的人,测试现在在我们的 git 中:

服务器:https://sourceforge.net/p/snapcpp/code/ci/master/tree/snapwebsites/tests/test_shutdown_server.cpp
客户:https://sourceforge.net/p/snapcpp/code/ci/master/tree/snapwebsites/tests/test_shutdown_client.cpp

这些使用了 snap 库,主要是依赖于 tcp 客户端/服务器的 snap_communicator:

TCP: https://sourceforge.net/p/snapcpp/code/ci/master/tree/snapwebsites/lib/tcp_client_server.cpp
通讯器:https://sourceforge.net/p/snapcpp/code/ci/master/tree/snapwebsites/lib/snap_communicator.cpp

最佳答案

正如您所发现的,套接字上的send 只会对要发送的数据进行排队。这实际上并不意味着服务器得到了它。这适用于阻塞和非阻塞套接字。

几种可能性:

  1. 确保在客户端程序退出之前在套接字上调用close。您没有在问题中说明是否发生了这种情况,但这可能是个好主意。

  2. 如果 #1 不起作用,请在套接字上使用 SO_LINGER 选项。适当设置超时间隔。

类似下面的内容

  struct linger ling;
  ling.l_onoff = 1;
  ling.l_linger = 3; // 3 second wait for data to finish being set.
  setsockopt(s, SOL_SOCKET, SO_LINGER, &ling, sizeof(ling));
  1. #2 的替代方法是修改您的协议(protocol),以便客户端在关闭套接字并退出之前从服务器获得某种确认消息。或者为了简单起见,客户端在退出之前等待服务器关闭套接字。 (recv 会在服务器关闭socket时返回0)

我的建议是确保您已实现#1。如果这不适合您,请评估#3。 #2,如果没有别的。

关于c - 在套接字上使用 O_NONBLOCK 后,有没有办法避免 HUP?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34113154/

相关文章:

c - 在函数结束时跳转到清理时 GOTO 是否被认为是无害的?

ruby-on-rails - 乘客的适当文件夹权限

linux - Nginx 在重定向时复制查询字符串

比较结构体的两个实例

c - 为什么我看不到执行 for 循环的混合线程

C - 大约 30 次读取后无法读取套接字

c++ - 如何以正确的方式读/写 UNIX 套接字?

java - SocketChannel write( ) 返回没有错误,但实际上没有发送数据

c - 段错误(核心转储)是什么意思? (快速排序代码)

linux - Git (1.8.1.2) 交互式添加时出错