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设置的超时时间无效呢

不要用read,改用recv()函数进行接收试试看吧。
如果实在没效果,把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()各使用场景)

setsockopt设置socket状态

linux下socket编程中setsockopt的作用

[python网络编程]利用socket编写简单的服务器

套接字编程 C++ setsockopt()