《Linux高性能服务器编程》阅读笔记:
Linux系统的3组IO复用系统调用(参考select()和poll()、epoll机制)都带有超时参数timeout,即它们不单能处理IO事件和信号(统一事件源),还可以处理定时器超时事件。但是需要注意两点:
(1)该系统调用既然能因为监听超时而返回,也就意味着能因为系统发来SIGALRM信号而返回。然而SIGALRM信号的产生可能是因为该系统调用超时,也可能是其它定时器触发的。对于前者,系统调用的返回值是0,对于后者,该系统调用的返回值是-1且errno等于EINTR(EINTR表示被中断)。
(2)IO复用可能在超时时间内因为有IO事件发生而返回,在这种情况下需要更新定时参数以反映剩余时间。
实现代码:
#define TIMEOUT 6000 //6000毫秒
int timeout = TIMEOUT;
time_t start, end;
while (1)
{
start = time(NULL);
int num = epoll_wait(epfd, events, MAX_EVENT_NUMBER, timeout);
//num < 0且errno==EINTR证明是该函数接收到其它程序块引起的SLGARM信号,则不退出
if ((num < 0) && (errno != EINTR))
{
perror("epoll_wait");
break;
}
if (num == 0)
{
timeout = TIMEOUT;
//处理超时事情
continue;
}
if (num > 0)
{
//执行到这里说明有IO时间发生
//处理IO事件
}
//计算超时时间的剩余量
end = time(NULL);
timeout -= (end - start) * 1000;
if (timeout <= 0)
{
//处理超时事情
timeout = TIMEOUT;
}
}