網絡通信中,很多操作會使得進程阻塞,這時我們要設定時間,到時間后強制返回,避免進程在沒有數據的情況下無限阻塞
這里我們總結一下網絡超時檢測的三種方法:
通過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設置的信號處理函數執行完后會重新執行被中斷的操作