使用select注意事项
使用select注意事项,监测的文件描述符一定要处理!
[待更新-] 为何会产生这种现象!
今天在实际使用timerfd 以及select过程中,发现FD_ISSET并不能有效检测文件描述符。实际现象就是一直可以进这个if。 后来终于发现,如果你不使用read等将文件描述符的内容进行读出,那么select函数会认为这个描述符始终处于置位状态,进而导致了这个现象。 下面的代码,可以实际的测试一下,重点观察注释掉read前后的差异。如果注掉read就会无限进入判断。
注意,初始超时时间,是指第一次到期的时间
#include <sys/timerfd.h>
#include <time.h>
#include <stdlib.h>
#include <stdio.h>
#include <stdint.h>
#include <sys/select.h>
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
int main()
{
int timer_fd;
uint64_t exp =0;
struct itimerspec itm;
fd_set r_set;
struct timespec time1 = {0, 0};
clock_gettime(CLOCK_REALTIME, &time1);
itm.it_interval.tv_sec = 10;
itm.it_interval.tv_nsec = 0;
itm.it_value.tv_sec = 10 + time1.tv_sec;
itm.it_value.tv_nsec = 0 + time1.tv_nsec;
timer_fd = timerfd_create(CLOCK_REALTIME, TFD_NONBLOCK);
timerfd_settime(timer_fd, TFD_TIMER_ABSTIME, &itm, NULL);
while(1){
FD_ZERO(&r_set);
FD_SET(timer_fd, &r_set);
FD_SET(1, &r_set);
select(timer_fd + 1, &r_set, NULL, NULL, NULL);
if(FD_ISSET(timer_fd, &r_set)){
printf("start!\n");
read(timer_fd, &exp, sizeof(uint64_t));
printf("Number of timeouts = %ld\n", exp);
sleep(1);
printf("end of if\n");
}
}
}