如何安全排出stdin?

时间:2019-09-30 06:45:40

标签: c stdin flush

我读了几个相同的问题,但答案却令我不满意。

他们认为最好的答案是:

while((ch = getchar()) != '\n') && ch != EOF);

如果我想在寻求答案之前先消耗标准输入,就会出现问题,例如:

#define DRAIN() while((ch = getchar()) != '\n') && ch != EOF)
int fun() {
    ...
    DRAIN();       //aaa, wipe other's ass before me
    ch = getchar();//bbb, ask my answer
    DRAIN();       //ccc, wipe myown ass
    ...
}

'aaa'的目的是清除先前代码留下的stdin中的字符,也许是在其他未知代码输入fun()之前。

但是,如果在'aaa'处,stdin中没有字符,则'aaa'会在此处阻止。

那什么是最好的选择?

=================

我在下面尝试过,没有效果。

#define DRAIN() do {                                        \
    char ch___ = 0;                                                \
    fd_set fds___;                                                 \
    struct timeval tv___;                                          \
    int ret___ = 0;                                                \
    FD_ZERO(&fds___);                                              \
    FD_SET(STDIN_FILENO, &fds___);                                 \
    tv___.tv_sec = 0;                                              \
    tv___.tv_usec = 5000;                                          \
    ret___ = select(STDIN_FILENO + 1, &fds___, NULL, NULL, &tv___);\
    printf("ret:%d\n", ret___);                                    \
    if (ret___ > 0) {                                              \
        while(getchar());                                          \
    }                                                              \
}while(0)

int main()
{
    printf("#");
    fflush(stdout);

    getchar();

    DRAIN();

    printf(">");
    fflush(stdout);
    printf(":%c\n", getchar());

    DRAIN();

    return 0;
}

$ ./a.out
#123
ret:0
>:2
ret:0

====第二次编辑====

看来这可行,我在Linux环境中:

#define DRAIN() do {                             \
    int flg___ = 0;                              \
    int flg___bak = 0;                           \
    char ch___ = 0;                              \
    flg___    = fcntl(STDIN_FILENO, F_GETFL, 0); \
    flg___bak = flg___;                          \
    flg___    |= O_NONBLOCK;                     \
    fcntl(STDIN_FILENO, F_SETFL, flg___);        \
    while((ch___ = getchar()) != '\n' && ch___ != EOF);  \
    fcntl(STDIN_FILENO, F_SETFL, flg___bak);     \
}while(0)

1 个答案:

答案 0 :(得分:0)

在DOS的早期,我使用以下顺序:

while ( kbhit() )
{
    ch = getch();
}
据我记得,

kbhit()是在conio.h中定义的,并且不可移植。但是,您可能很幸运能够访问它。