1. 根据errno和recv结果进行判断
在unix/linux下,非阻塞模式socket可以采用recv msg_peek的方式进行判断,其中msg_peek保证了仅仅进行状态判断,而不影响数据接收
对于主动关闭的socket, recv返回-1,而且errno被置为9(#define ebadf 9 /* bad file number */)或104 (#define econnreset 104 /* connection reset by peer */)
对于被动关闭的socket,recv返回0,而且errno被置为11(#define ewouldblock eagain /* operation would block */)
对正常的socket, 如果有接收数据,则返回>0, 否则返回-1,而且errno被置为11(#define ewouldblock eagain /* operation would block */)
因此对于简单的状态判断(不过多考虑异常情况):
- recv返回>0, 正常
- 返回-1,而且errno被置为11 正常
- 其它情况 关闭
int np_socket_alive (int conn_fd) { char buff[32]; int recv_buff = recv (conn_fd, buff, sizeof (buff), msg_peek); int sockerr = errno; if (recv_buff > 0) // get data return 1; if ((recv_buff == -1) && (sockerr == ewouldblock)) // no data recieved return 1; return -1; }
2. 通过tcp_keepalive进行判断