与select()混淆 - stdout从未准备好写入

时间:2015-08-28 08:36:10

标签: c unix select io

这是一个简单的select()循环:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <sys/select.h>

#define BUFSIZE 999

int main()
{
    int            select_result;
    fd_set         read_fds, write_fds;
    struct timeval timeout = {0, 400000}, timeoutcopy;
    const int max_fd = STDIN_FILENO > STDOUT_FILENO ? STDIN_FILENO :
                                                      STDOUT_FILENO;
    char buffer[BUFSIZE];

    fcntl(STDIN_FILENO, F_SETFL, fcntl(STDIN_FILENO, F_GETFL) | O_NONBLOCK);
    fcntl(STDOUT_FILENO, F_SETFL, fcntl(STDOUT_FILENO, F_GETFL) | O_NONBLOCK);
    printf("Enter loop\n");
    while(1) {
        FD_ZERO(&read_fds);
        FD_ZERO(&write_fds);
        printf("Loop\n");
        FD_SET(STDIN_FILENO, &read_fds);
        FD_SET(STDOUT_FILENO, &write_fds);

        timeoutcopy = timeout;
        if ((select_result = select(max_fd, &read_fds, &write_fds, NULL,
                                    &timeoutcopy)) < 0) {
            return select_result;
        }

        if (FD_ISSET(STDIN_FILENO, &read_fds))
            printf("Stdin ready for read\n");
            fgets(buffer, BUFSIZE, stdin);
            if (strlen(buffer))
                printf("Stdin content: %s\n", buffer);
        if (FD_ISSET(STDOUT_FILENO, &write_fds))
            printf("Stdout ready for write\n");
    }
}

它只使用stdin轮询stdoutselect(),超时为400000毫秒。当stdin准备就绪时,它会尝试读取内容并打印出来。当stdout准备就绪时,它只是打印,准备就绪。

出于某种原因,select()来电stdin之后就没有准备好,为什么?

2 个答案:

答案 0 :(得分:2)

你的max_fd应该是&#34;三组中任何一组中编号最高的文件描述符加上1。&#34;根据选择的手册页。我应该重命名它,以便你记得加1。

答案 1 :(得分:0)

max_fd必须加1。名称确实令人困惑,&#34; nfds&#34;可能更清楚了。请看这里:

select man page