10

这是来自“man select”的代码示例加上几行来读取正在写入的实际文件。我怀疑当./myfile.txt被写入时,select会返回它现在可以从该 fd 中读取。但是,只要 txt 文件存在,select 就会在 while 循环中不断返回。我希望它仅在将新数据写入文件末尾时返回。我认为这就是它应该如何工作的。

#include <stdio.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>

int
main(void)
{
    fd_set rfds;
    struct timeval tv;
    int retval;

    int fd_file = open("/home/myfile.txt", O_RDONLY);

   /* Watch stdin (fd 0) to see when it has input. */
    FD_ZERO(&rfds);
    FD_SET(0, &rfds);
    FD_SET(fd_file, &rfds);

   /* Wait up to five seconds. */
    tv.tv_sec = 5;
    tv.tv_usec = 0;

   while (1)
   {
     retval = select(fd_file+1, &rfds, NULL, NULL, &tv);
     /* Don't rely on the value of tv now! */

     if (retval == -1)
        perror("select()");
     else if (retval)
        printf("Data is available now.\n");
        /* FD_ISSET(0, &rfds) will be true. */
     else
        printf("No data within five seconds.\n");
   }

   exit(EXIT_SUCCESS);
}
4

1 回答 1

17

磁盘文件始终准备好读取(但如果您已经在文件末尾,则读取可能会返回 0 字节),因此您无法select()在磁盘文件上使用来了解何时将新数据添加到文件中。

POSIX 说:

与常规文件关联的文件描述符应始终为准备读取、准备写入和错误条件选择 true。

此外,正如cnicutar在现已删除的帖子中指出的那样,通常,您必须在每次迭代时初始化 FD_SET。在您的代码中,您正在监视一个 fd,并且该 fd 始终准备就绪,因此 FD_SET 实际上并没有改变。但是,如果您有 5 个描述符要监视,并且select检测到只有一个已准备好,那么在下一次迭代中,将仅监视一个描述符(除非您重置 FD_SET)。这使得使用select变得棘手。

于 2012-08-10T12:57:41.987 回答