网络超时检测

  • Post author:
  • Post category:其他


1. 网络超时

在网络通信中。很多操作会使进程阻塞,如TCP套接字中的recv/accept/connect,如UDP套接字中的recvfrom。为了避免进程在没有数据时无限制的阻塞,以下有三种方法设置网络超时检测。

2. 网络超时检测三种方法

1)法1

设置socket的属性SO_RCVTIMEO

示例代码如下:

struct timeval tv;

tv.tv_sec = 5; //设置5秒时间

tv.tv_usec = 0;

setsocket(sockfd, SOL_SOCKET, SO_RCVTIMEO,&tv, sizeof(tv));//设置超时

recv() / recvfrom() //从socket读取数据,注:如果recv一直收不到数据,5s之后recv会返回 -1,并置errno = EAGAIN。

2)法2

用select函数检测socket是否‘ready’

示例代码如下:

struct fd_set rdfs;

struct timeval tc = {5, 0}; //设置5秒时间

FD_ZERO(&rdfs);

FD_SET(sockfd, &rdfs);

If(select(sockfd + 1, &rdfs, NULL,NULL, &tv) <= 0) //检测sockfd就绪,注:select如果一直不触发,5s之后会返回 0,返回错误信息Success。

{

perror(“”);

return;

}

3)法3

设置定时器timer,捕捉SIGALRM信号

示例代码如下:

void handler(int signo)

{

printf(“5s has passed!\n”)

return;

}

struct sigaction act;

sigaction(SIGALRM, NULL, &act);

act.sa_handler = handler;

act.sa_flag &= ~SA_RESTART;

sigaction(SIGALRM, &act, NULL);

Alarm(5); //进程在5秒后没有收到数据,定时器就会发出SIGALRM信号给进程,进程收到这个信号后,执行完信号处理函数,阻塞函数就会直接返回,不会阻塞。

If(recv(,,,) < 0)// 注:如果recv一直收不到数据,5s之后recv会返回-1,返回错误消息类型为Interrupted system call。



版权声明:本文为weixin_42048417原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。