C语言socket编程中setsockopt设置超时时间对read无效
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C语言socket编程中setsockopt设置超时时间对read无效相关的知识,希望对你有一定的参考价值。
struct timeval ti;
ti.tv_sec = 2;
ti.tv_usec = 0;
//定义超时时间
if (setsockopt(fd,SOL_SOCKET,SO_RCVTIMEO,&ti,sizeof(ti)) == -1)
cout << "setsockopt error" <<endl;
close(fd);
return -1;
//使用read接收
while(read(fd,&ch,1))
recvstr << ch;
使用中发现read依旧阻塞直到对方服务器关闭连接,这是为什么。使用read阻塞的时候我该如何关闭套接字?为什么setsockopt设置的超时时间无效呢
如果实在没效果,把socket一开始就设置为非阻塞模式吧:
u_long mode = 1;
ioctlsocket(fd,FIONBIO,&mode);追问
对方服务器发送字节数比较多,已经分包了。recv只是收一个包?
追答recv和read都不会检查包是否完整,最好不要依赖这种函数去认为数据收好了,即使发送方没有分包只发10个字节,由于网络原因什么的接收方也有可能两次才收到,例如第一次收到8个,第二次收到2个。一个包被拆成多个发送,接收方应该以包里的特殊字符为根据去判断是否接收完毕。例如,假设跟发送方约定一个完整数据包的结尾会有一个回车符,那么你在收到前几个分包点时候就检查是否收到结尾的回车符了,没有收到的话就继续等待下一个分包。
参考技术A Windows 环境下:定义:int nNetTimeout=1000;//1 秒
//设置接收超时
setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char*)&nNetTimeout,sizeof(int));
Linux 环境下:
定义:struct timeval timeout = 3,0;
//设置接收超时
setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char*)&timeout,sizeof(struct timeval));
linux下socket编程中setsockopt的作用
如题所示,在linux进行socket编程的时候,一般而言,socket,bind,listen三步曲之后,就开始接收客户端请求,然后实现收发数据。
如下所示的代码,是没有setsockopt的情况:
#include<stdlib.h>
#include<stdio.h>
#include<string.h>
#include<errno.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<unistd.h>
#define MAXLINE 4096
int main(int argc,char** argv)
int listenfd,connfd;
struct sockaddr_in servaddr;
char buff[4096];
int n;
if((listenfd=socket(AF_INET,SOCK_STREAM,0))==-1)
printf("create socket error: %s(errno: %d)\\n",strerror(errno),errno);
return 0;
memset(&servaddr,0,sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port = htons(6666);
if(bind(listenfd,(struct sockaddr*)&servaddr,sizeof(servaddr))==-1)
printf("bind socket error: %s(errno: %d)\\n",strerror(errno),errno);
return 0;
if(listen(listenfd,10)==-1)
printf("listen socket error: %s(errno: %d)\\n",strerror(errno),errno);
return 0;
printf("====== waiting for client's request ======\\n");
while(1)
if((connfd=accept(listenfd,(struct sockaddr*)NULL,NULL))==-1)
printf("accept socket error: %s(errno: %d)\\n",strerror(errno),errno);
continue;
n = recv(connfd,buff,MAXLINE,0);
buff[n] = '\\0';
printf("recv msg from client: %s\\n",buff);
close(connfd);
close(listenfd);
return 0;
运行这个代码,然后模拟客户端接入,最后终止程序。这时候通过netstat命令查看系统tcp端口占用情况,发现6666端口仍然处于活跃状态,只是state变为了TIME_WAIT。
这时候,如果再次运行./server,会提示绑定失败,端口已经在使用。
这种情形不会一直持续,等待一会就好了。
今天介绍的setsockopt函数就是来解决这个问题的。
我们在bind()之前,调用这个函数设置端口重复使用。
这时候,重复上面的操作:
当我们关闭服务端,再次开启的时候,就不会出现端口已经被占用的错误了。
其实这个问题,在一般的开发中并不会出现,只是我们频繁做测试的时候,这个监听端口才会出现不能绑定,需要等待一段时间的情况。但是使用了setsockopt之后,相当于强行解除了原来的绑定。是一种保险的做法。
最后给出添加了setsockopt函数的代码:
#include<stdlib.h>
#include<stdio.h>
#include<string.h>
#include<errno.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<unistd.h>
#define MAXLINE 4096
int main(int argc,char** argv)
int listenfd,connfd;
struct sockaddr_in servaddr;
char buff[4096];
int n;
if((listenfd=socket(AF_INET,SOCK_STREAM,0))==-1)
printf("create socket error: %s(errno: %d)\\n",strerror(errno),errno);
return 0;
memset(&servaddr,0,sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port = htons(6666);
int on = 1;
setsockopt(listenfd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on));
if(bind(listenfd,(struct sockaddr*)&servaddr,sizeof(servaddr))==-1)
printf("bind socket error: %s(errno: %d)\\n",strerror(errno),errno);
return 0;
if(listen(listenfd,10)==-1)
printf("listen socket error: %s(errno: %d)\\n",strerror(errno),errno);
return 0;
printf("====== waiting for client's request ======\\n");
while(1)
if((connfd=accept(listenfd,(struct sockaddr*)NULL,NULL))==-1)
printf("accept socket error: %s(errno: %d)\\n",strerror(errno),errno);
continue;
n = recv(connfd,buff,MAXLINE,0);
buff[n] = '\\0';
printf("recv msg from client: %s\\n",buff);
close(connfd);
close(listenfd);
return 0;
setsockopt函数并不是必须的,但是为了保险起见,最好是加上。
以上是关于C语言socket编程中setsockopt设置超时时间对read无效的主要内容,如果未能解决你的问题,请参考以下文章
C语言socket服务端报错:bind socket error: Address already in use(errno: 98)(setsockopt()各使用场景)
C语言socket服务端报错:bind socket error: Address already in use(errno: 98)(setsockopt()各使用场景)