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

Решение

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

6

Другие решения

Если ваш Linux не имеет conio.h что поддерживает kbhit() ты можешь посмотреть здесь для кода Моргана Мэттьюса предоставлять kbhit() функциональность, совместимая с любой POSIX-совместимой системой.

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

10

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

#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

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

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

2
По вопросам рекламы [email protected]