我有一个应用程序需要在某个时间向服务器发送一些数据。简单的方法是关闭连接,然后当我想发送东西时再次打开它。但我想保持连接打开,所以当我想发送数据时,我首先使用此函数检查连接:
bool is_connected(int sock)
{
unsigned char buf;
int err = recv(sock,&buf,1,MSG_PEEK);
return err == -1 ? false : true;
}
不好的是,这不起作用。当没有数据要接收时它会挂起。我能做什么?如何检查连接是否仍处于打开状态?
答案 0 :(得分:23)
请先检查然后再发送。这是浪费的努力,无论如何都无法工作 - 状态可以在您检查和发送时间之间发生变化。只要做你想做的事情,如果失败就处理错误。
要检查状态,请使用:
int error_code;
int error_code_size = sizeof(error_code);
getsockopt(socket_fd, SOL_SOCKET, SO_ERROR, &error_code, &error_code_size);
答案 1 :(得分:12)
您需要使用O_NONBLOCK
设置fcntl
来启用非阻止行为。一种简单但非标准的非阻塞读取方法是使用:
recv(sock, &buf, 1, MSG_PEEK | MSG_DONTWAIT);
之后,必须检查errno是否失败。它可能会因EAGAIN
而失败,或者可能因EBADF
或ENOTCONN
等而失败。
显然,处理这个问题最简单,最干净的方法是避免“忘记”套接字是否连接。一旦recv
返回0或send
返回EPIPE
,您就会注意到套接字是否已断开连接。
答案 2 :(得分:3)
TCP的默认使用不允许非常及时地检测死套接字(在正常闭包之外),所以我建议像这样的“is_connected”函数对于所有实际目的来说都是无用的。考虑实施一个应用层保持活动,并根据及时响应(或缺乏响应)跟踪它是否存活。
编辑:发布后我看到了BoBTFish的链接,这实际上是一回事。