26

我有一个应用程序需要在某个时间向服务器发送一些数据。简单的方法是关闭连接,然后在我想发送一些东西时再次打开它。但我想保持连接打开,所以当我想发送数据时,我首先使用这个函数检查连接:

bool is_connected(int sock)
{
    unsigned char buf;
    int err = recv(sock,&buf,1,MSG_PEEK);
    return err == -1 ? false : true;
}

不好的部分是这不起作用。当没有数据要接收时它会挂起。我能做些什么?如何检查连接是否仍然打开?

4

3 回答 3

31

不要先检查然后发送。这是浪费精力并且无论如何都不会工作 - 状态可能会在您检查和发送之间发生变化。只要做你想做的事,如果失败就处理错误。

要检查状态,请使用:

int error_code;
int error_code_size = sizeof(error_code);
getsockopt(socket_fd, SOL_SOCKET, SO_ERROR, &error_code, &error_code_size);
于 2012-09-13T17:38:04.327 回答
13

O_NONBLOCK您需要通过设置using来启用非阻塞行为fcntl。进行非阻塞读取的一种简单但非标准的方法是使用:

recv(sock, &buf, 1, MSG_PEEK | MSG_DONTWAIT);

之后,如果失败,您必须检查 errno。它可以失败,EAGAIN也可以失败,EBADF等等ENOTCONN


显然,处理这个问题的最简单和最干净的方法是避免“忘记”套接字是否连接。您会注意到一旦 arecv返回 0 或 asend返回,套接字是否断开连接EPIPE

于 2012-09-13T08:47:27.920 回答
4

TCP 的默认使用不允许非常及时地检测死套接字(在正常关闭之外),所以我建议像这样的“is_connected”函数对于所有实际目的基本上是无用的。考虑实施应用层保持活动,并根据及时响应(或缺乏响应)跟踪它是否处于活动状态。

编辑:发布后我看到 BoBTFish 的链接,这实际上是一回事。

于 2012-09-13T17:34:41.520 回答