Linux,套接字,非阻塞连接

her*_*ver 14 c sockets linux asynchronous epoll

我想创建一个非阻塞连接.像这样:

socket.connect(); // returns immediately
Run Code Online (Sandbox Code Playgroud)

为此,我使用另一个线程,一个无限循环和Linux epoll.像这样(伪代码):

// in another thread
{
  create_non_block_socket();
  connect();

  epoll_create();
  epoll_ctl(); // subscribe socket to all events
  while (true)
  {
    epoll_wait(); // wait a small time(~100 ms)
    check_socket(); // check on EPOLLOUT event
  }
}
Run Code Online (Sandbox Code Playgroud)

如果我运行服务器然后运行客户端,那么一切正常.如果我第一次运行客户端,等待一小段时间,运行服务器,然后客户端不连接.

我究竟做错了什么?也许它可以做得与众不同?

nos*_*sid 36

您应该使用以下步骤进行异步连接:

  • 用.创建套接字 socket(..., SOCK_NONBLOCK, ...)
  • 开始连接 connect(fd, ...)
  • 如果返回值既不是也不0EINPROGRESS,则中止并出错
  • 等到fd发信号为准备输出
  • 检查套接字的状态 getsockopt(fd, SOL_SOCKET, SO_ERROR, ...)
  • DONE

没有循环 - 除非你想处理EINTR.

如果首先启动客户端,则应ECONNREFUSED在最后一步中看到错误.如果发生这种情况,请关闭套接字并从头开始.

如果没有看到更多详细信息,很难说出代码有什么问题.我想,你不会因check_socket操作错误而中止.

  • 当`getsockopt(fd,SOL_SOCKET,SO_ERROR,...)`返回0,so_error为0时,这并不意味着套接字已连接。这意味着到目前为止,没有发生任何错误。在这种情况下,您需要调用`getpeername()`,如果`getpeername()`返回0,则表示套接字已连接。如果没有连接套接字,`getpeername()`在errno中以ENOTCONN返回-1。getsockopt(fd,SOL_SOCKET,SO_ERROR,...)可以通知您有关拒绝连接的信息,但不能通知您有关已连接的套接字的信息。您需要使用`getpeername()`或其他方法来确保套接字已连接。 (2认同)

小智 7

有几种方法可以测试非阻塞连接是否成功。

  1. 首先调用 getpeername(),如果失败并出现错误 ENOTCONN,则连接失败。然后使用 SO_ERROR 调用 getsockopt 以获取套接字上的未决错误
  2. 调用长度为0的read,如果读取失败,则连接失败,read的errno表示连接失败的原因;如果连接成功,则 read 返回 0
  3. 再次调用connect;如果 errno 是 EISCONN,则连接已经连接并且第一次连接成功。

参考:UNIX 网络编程 V1