Использование 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?

8
30 марта '15 в 1:36
источник поделиться
4 ответов

Процитированные выше упоминания могут быть полезными. Вот пример, иллюстрирующий использование 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 будут использовать тот же тип данных для bool ncurses, что и компилятор C++, используемый для настройки ncurses.

6
30 марта '15 в 2:07
источник

Связанные вопросы


Похожие вопросы

Если ваш linux не имеет conio.h который поддерживает kbhit() вы можете посмотреть здесь код Morgan Mattews, чтобы обеспечить функциональность kbhit() совместимую с любой совместимой с POSIX системой.

Поскольку трюк дезактивирует буферизацию на уровне termios, он также должен решить проблему getchar() как показано здесь.

10
30 марта '15 в 1:40
источник

Компактное решение, основанное на Кристофе,

#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, терминал остается в странном состоянии. (Вздох)

5
13 июля '17 в 0:11
источник

Хотя использование ncurses функционально эквивалентно API Turbo C "conio.h", более полным решением является использование реализации conio, как можно найти здесь.

Вы загружаете и используете его в своей программе для очень полной реализации интерфейса conio в Linux. (Или OSX.) Написал Рон Берки.

2
30 марта '15 в 2:13
источник

Посмотрите другие вопросы по меткам или Задайте вопрос