一、使用recv()+errno
1、正常recv()返回读取的长度
2、没有数据时recv()返回-1,errno等于11 errno == EINTR
3、断开连接的时候,recv()返回0
二、使用getsock()
#include<netinet/in.h>
#include<sys/time.h>
#include<arpa/inet.h>
#include<netinet/tcp.h>
int SocketConnected(int sock)
{
struct tcp_info info;
int len = sizeof(info);
if( sock <= 0 )
{
printf("socket %d\n", sock);
return 0;
}
getsockopt(sock, IPPROTO_TCP, TCP_INFO, &info, (socklen_t *)&len);
if((info.tcpi_state == TCP_ESTABLISHED))
{
//printf("socket connected\n");
return 1;
}
else
{
printf("socket disconnected\n");
return 0;
}
}
在需要检测的地方调用SocketConnected()
ret = SocketConnected(cur_port.TcpclientHandle[RS232]);
if(ret == 0)
{
//重连---
}
注意:
1、当服务端检测是否断开填入的是当前连接的客户端fd
2、当客户端检测是否断开填入的是当前客户端的fd
三、SO_KEEPALIVE 机制
这是socket库提供的功能,设置接口是setsockopt API
原理:为socket设置了KEEPALIVE选项,TCP/IP栈在检测到对方掉线后, 任何在该socket上进行的调用(发送/接受调用)就会立刻返回,错误号是WSAENETRESET ;同时,此后的任何在该socket句柄的调用会立刻失败,并返回WSAENOTCONN错误。
BOOL bSet=TRUE;
setsockopt(hSocket,SOL_SOCKET,SO_KEEPALIVE,(const char*)&bSet,sizeof(BOOL));
缺点:
默认设置是空闲2小时才发送一个“保持存活探测分节”,不能保证实时检测!
当然也可以修改时间间隔参数,但是会影响到所有打开此选项的套接口!
关联了完成端口的socket可能会忽略掉该套接字选项。
四、应用层增加心跳机制
在应用层定时发送固定数据给服务器,服务器收不到心跳数据则为断开
缺点:增加应用开销,影响应用层协议
一般不建议使用