在Linux上使用kbhit()和getch()

Box*_*iom 8 c++ linux getch conio kbhit

在Windows上,我有以下代码来查找输入而不中断循环:

#include <conio.h>
#include <Windows.h>
#include <iostream>

int main()
{
    while (true)
    {
        if (_kbhit())
        {
            if (_getch() == 'g')
            {
                std::cout << "You pressed G" << std::endl;
            }
        }
        Sleep(500);
        std::cout << "Running" << std::endl;
    }
}
Run Code Online (Sandbox Code Playgroud)

但是,看到没有conio.h,在Linux上实现这一点的最简单方法是什么?

Chr*_*phe 12

如果您的Linux没有conio.h支持,kbhit()您可以在这里查看Morgan Mattews的代码,kbhit() 与任何POSIX兼容系统兼容的方式提供功能.

由于技巧在termios级别停用缓冲,它也应解决此处所示getchar()问题.

  • 这个答案解决了我的问题,所以感谢Christophe.但我想补充说,由于使用了FIONREAD,我需要"#include <sys/ioctl.h>"才能让Morgan McGuire的功能在我的程序中运行,并且没有明确列出找到他的代码的网页. (4认同)
  • stropts.h 没有这样的文件或目录。显然linux不再支持这个了 (2认同)

Tho*_*key 7

如上所述的ncurses howto可能会有所帮助.这是一个示例,说明如何使用ncurses像conio示例:

#include <ncurses.h>

int
main()
{
    initscr();
    cbreak();
    noecho();
    scrollok(stdscr, TRUE);
    nodelay(stdscr, TRUE);
    while (true) {
        if (getch() == 'g') {
            printw("You pressed G\n");
        }
        napms(500);
        printw("Running\n");
    }
}
Run Code Online (Sandbox Code Playgroud)

请注意,使用ncurses时,iostream不使用标头.这是因为将stdio与ncurses混合会产生意想不到的结果.

顺便说一句,ncurses定义TRUEFALSE.正确配置的ncurses将使用与bool用于配置ncurses的C++编译器相同的ncurses数据类型.


PBS*_*PBS 6

基于Christophe答案的紧凑型解决方案是

#include <sys/ioctl.h>
#include <termios.h>

bool kbhit()
{
    termios term;
    tcgetattr(0, &term);

    termios term2 = term;
    term2.c_lflag &= ~ICANON;
    tcsetattr(0, TCSANOW, &term2);

    int byteswaiting;
    ioctl(0, FIONREAD, &byteswaiting);

    tcsetattr(0, TCSANOW, &term);

    return byteswaiting > 0;
}
Run Code Online (Sandbox Code Playgroud)

与该答案不同,在程序退出后,这不会使终端处于奇怪的状态.但是,它仍然将字符留在输入缓冲区中,因此按下的键将不受欢迎地出现在下一个提示行中.

解决此问题的另一种解决方案是

void enable_raw_mode()
{
    termios term;
    tcgetattr(0, &term);
    term.c_lflag &= ~(ICANON | ECHO); // Disable echo as well
    tcsetattr(0, TCSANOW, &term);
}

void disable_raw_mode()
{
    termios term;
    tcgetattr(0, &term);
    term.c_lflag |= ICANON | ECHO;
    tcsetattr(0, TCSANOW, &term);
}

bool kbhit()
{
    int byteswaiting;
    ioctl(0, FIONREAD, &byteswaiting);
    return byteswaiting > 0;
}
Run Code Online (Sandbox Code Playgroud)

用法如下

enable_raw_mode();
// ...
if (kbhit()) ...
// ...
disable_raw_mode();
tcflush(0, TCIFLUSH); // Clear stdin to prevent characters appearing on prompt
Run Code Online (Sandbox Code Playgroud)

现在,在执行第一行和最后一行之间键入的任何字符都不会显示在终端中.但是,如果您使用Ctrl + C退出,则终端处于奇怪的状态.(叹)