Использование kbhit () и getch () в Linux
в 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;
    }
}
однако, видя, что нет conio.h, каков самый простой способ достижения этого же самого в Linux?
4 ответов
приведенное выше 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");
    }
}
обратите внимание, что с ncurses, то iostream заголовок не используется.  Это потому, что смешивание stdio с ncurses может иметь неожиданные результаты.
ncurses, кстати, определяет TRUE и FALSE.  Правильно настроенные ncurses будут использовать тот же тип данных для ncurses' bool как компилятор C++, используемый для настройки на ncurses.
если ваш linux не имеет conio.h, обслуживающей kbhit() вы можете ознакомиться вот код Моргана Мэттьюса обеспечить kbhit() функциональность в пути совместимом с любой системой POSIX уступчивой.
как трюк деактивации буферизации на уровне termios, он также должен решить getchar() проблема, как показал здесь. 
при использовании ncurses функционально эквивалентно Turbo C " conio.H " API, более полное решение-использовать реализацию conio, как может быть найти здесь.
вы загружаете и используете его в своей программе для очень полной реализации интерфейса conio в Linux. (Или OSX.) Автор Рон Берки.
компактное решение, основанное на ответе Кристофа, -
#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;
}
В отличие от этого ответа, это не оставит терминал в странном состоянии после выхода программы. Однако он по-прежнему оставляет символы, сидящие во входном буфере, поэтому нажатая клавиша будет неприятно отображаться в следующей строке приглашения.
другое решение, которое устраняет эту проблему
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;
}
использование выглядит следующим образом
enable_raw_mode();
// ...
if (kbhit()) ...
// ...
disable_raw_mode();
tcflush(0, TCIFLUSH); // Clear stdin to prevent characters appearing on prompt
Теперь все символы набраны между выполнением первой и последней строк в терминале не будет отображаться. Однако, если вы выходите с Ctrl+C, терминал и ушел в странном состоянии. (Вздох)