一,网络超时的原因
在网络通信中,有很多操作会使得进程阻塞,超时检测的必要性就体现了出来
避免进程在没有数据时无限制地阻塞,设定的时间到时,进程从原操作返回继续运行。
二、网络超时的解决办法
(1)设置socket的属性
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;
}
}
(2)select函数检测socket是够准备
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;
}
。。。。。。
}
(3)设置定时器timer 捕捉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设置的信号处理函数执行完后会重新执行被中断的操作。