如何使用select()在C中读取键盘输入

dru*_*rum 9 c keyboard stdin textinput select-function

我试图使用select()来读取键盘输入,我陷入了困境,我不知道如何从键盘读取并使用文件描述符这样做.我被告知使用STDIN和STDIN_FILENO来解决这个问题,但我仍然感到困惑.
我该怎么做?

lus*_*oog 5

你的问题听起来有点困惑.select()用于阻止输入可用.但是,你有正常的文件读取功能(如实际读数read,fread,fgetc,等).

这是一个简单的例子.它会阻塞,直到stdin至少有一个字符可供阅读.但是当然除非你将终端更改为某种未烹饪模式,否则它会一直阻塞,直到您按下回车键,此时键入的任何字符都会被刷新到文件缓冲区(来自某个终端缓冲区).

#include <stdio.h>
#include <sys/select.h>

int main(void) {
    fd_set s_rd, s_wr, s_ex;
    FD_ZERO(&s_rd);
    FD_ZERO(&s_wr);
    FD_ZERO(&s_ex);
    FD_SET(fileno(stdin), &s_rd);
    select(fileno(stdin)+1, &s_rd, &s_wr, &s_ex, NULL);
    return 0;
}
Run Code Online (Sandbox Code Playgroud)


jwa*_*zko 5

如前所述,通过使用select,您可以仅监视例如stdin来检查输入数据是否已可供读取。如果可用,则可以使用例如fgets安全地将输入数据读取到某个缓冲区,如下所示:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int main(int argc, char *argv[])
{
    fd_set rfds;
    struct timeval tv;
    int retval, len;
    char buff[255] = {0};

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

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

    retval = select(1, &rfds, NULL, NULL, &tv);

    if (retval == -1){
        perror("select()");
        exit(EXIT_FAILURE);
    }
    else if (retval){
        /* FD_ISSET(0, &rfds) is true so input is available now. */

        /* Read data from stdin using fgets. */
        fgets(buff, sizeof(buff), stdin);

        /* Remove trailing newline character from the input buffer if needed. */
        len = strlen(buff) - 1;
        if (buff[len] == '\n')
            buff[len] = '\0';

        printf("'%s' was read from stdin.\n", buff);
    }
    else
        printf("No data within five seconds.\n");            

    exit(EXIT_SUCCESS);
}
Run Code Online (Sandbox Code Playgroud)