網(wǎng)絡通信中,很多操作會使得進程阻塞,這時我們要設定時間,到時間后強制返回,避免進程在沒有數(shù)據(jù)的情況下無限阻塞
這里我們總結一下網(wǎng)絡超時檢測的三種方法:
通過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函數(shù)的一個參數(shù)實現(xiàn)超時處理
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信號的處理函數(shù)
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秒鐘內(nèi)沒有數(shù)據(jù)到來,內(nèi)核產(chǎn)生SIGALRM信號中斷當前操作。我們知道設置信號捕捉函數(shù)可以用signal函數(shù)或是sigaction函數(shù)。但這里只能使用sigaction函數(shù),因為signal設置的信號處理函數(shù)執(zhí)行完后會重新執(zhí)行被中斷的操作