网络通信中,很多操作会使得进程阻塞,这时我们要设定时间,到时间后强制返回,避免进程在没有数据的情况下无限阻塞
这里我们总结一下网络超时检测的三种方法:
一、通过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\ 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设置的信号处理函数执行完后会重新执行被中断的操作
因篇幅问题不能全部显示,请点此查看更多更全内容