Linux C recv函数阻塞问题

问题描述:

有个Linux项目用到TCP/IP通讯,但是recv()函数老是阻塞,没有返回;setsockopt()也设置了超时时间,同时在执行recv()之前也增加了select()查看套接字的状态,如果状态异常异常是不会执行recv()函数的


原因分析:

1、recv()是Linux的系统函数,肯定不会卡死,除非Linux内核故障,基本可以排除;
2、客户端向服务器端发送数据,服务器端没有接收,导致缓冲区满,客户端无法接收数据,因此客户端阻塞?等等,我不是加了超时吗?怎么还会阻塞?
3、读写超时设置有问题?还真是,其他函数都做了返回值判断,唯独读写超时没有判断返回值,因此没及时发现问题;读写超时调用的函数时setsockopt(),返回-1,执行失败了,所以recv()一直阻塞

解决方案:

setsockopt() 函数原型如下:

int setsockopt(int sockfd, int level, int optname,const void *optval, socklen_t optlen);

sockfd:标识一个套接口的描述字。
level:选项定义的层次;支持SOL_SOCKET、IPPROTO_TCP、IPPROTO_IP和IPPROTO_IPV6。
optname:需设置的选项。
optval:指针,指向存放选项待设置的新值的缓冲区。
optlen:optval缓冲区长度。

以下是我的使用方式:

int nTimeout=1000;//1秒
// 发送超时
setsockopt(socket,SOL_S0CKET, SO_SNDTIMEO, (char *)&nNetTimeout,sizeof(int));
// 接收超时
setsockopt(socket,SOL_S0CKET, SO_RCVTIMEO, (char *)&nNetTimeout,sizeof(int));

上面是 Windows 的用法,对于设置 SO_SNDTIMEO 和 SO_RCVTIMEO,Windows环境下只能传 DWORD;Linux 环境下, 只能传结构体 struct timeval,传 int 会失败,须如下使用:

struct timeval timeout = {1, 0}; // 分别是秒和毫秒
 // 发送超时
setsockopt(socket,SOL_SOCKET,SO_SNDTIMEO,(char *)&timeout,sizeof(struct timeval));
// 接收超时
setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char *)&timeout,sizeof(struct timeval));

你可能感兴趣的:(C/C++,BUG,Linux,c++)