使用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");
        }
    }
}