网络超时检测的三种方法
作者:于老师,讲师。
网络通信中,很多操作会使得进程阻塞,这时我们要设定时间,到时间后强制返回,避免进程在没有数据的情况下无限阻塞
这里我们总结一下网络超时检测的三种方法:
通过setsockopt设置套接字属性SO_RCVTIMEO
struct timeval t = {5, 0}
if (setsockopt(listenfd, SOL_SOCKET, SO_RCVTIMEO, &t, sizeof(t)) == -1) { perror("setsockopt"); return -1; } memset(&peeraddr, 0, sizeof(peeraddr)); len = sizeof(peeraddr); if ((connfd = accept(listenfd, (struct sockaddr *)&peeraddr, &len)) == -1) { printf("errno=%d: %s\n", errno, strerror(errno)); if (errno == EAGAIN) { printf("timeout\n"); return -1; } }二、设定select函数的一个参数实现超时处理
struct timeval t= {3, 0};
while (1) { 。。。。。。 t.tv_sec = 3; t.tv_usec = 0; if ((ret = select(maxfd+1, &rdfs, NULL, NULL, &t)) == -1) { perror("select"); return -1; } 。。。。。。 }三、设定一个定时器捕捉SIGALRM信号做超时控制
struct sigaction act;
sigaction(SIGALRM, NULL, &act); //获取SIGALRM信号的属性 act.sa_handler = handler; // 设置SIGALRM信号的处理函数 sigaction(SIGALRM, &act, NULL); // 设置SIGALRM信号的属性 alarm(3); // 定时器设置3秒钟 while (1) { if ((connfd = accept(listenfd, (struct sockaddr *)&peeraddr, &len)) == -1) { if (errno == EINTR) { printf("timeout\n"); return -1; } }定时器3秒钟内没有数据到来,内核产生SIGALRM信号中断当前操作。我们知道设置信号捕捉函数可以用signal函数或是sigaction函数。但这里只能使用sigaction函数,因为signal设置的信号处理函数执行完后会重新执行被中断的操作