Jak już wspomniano, możesz użyć sigaction
do zalewkowania ctrl-c lub select
do zalewkowania dowolnego standardowego wejścia.
Zwróć jednak uwagę, że w przypadku tej drugiej metody musisz również ustawić TTY tak, aby był w trybie znak po czasie, a nie wiersz po czasie. To drugie jest domyślne - jeśli wpiszesz wiersz tekstu, nie zostanie on wysłany na stdin uruchomionego programu, dopóki nie naciśniesz klawisza Enter.
Będziesz musiał użyć tej tcsetattr()
funkcji, aby wyłączyć tryb ICANON i prawdopodobnie również wyłączyć ECHO. Z pamięci musisz także ustawić terminal z powrotem w tryb ICANON, gdy program zakończy pracę!
Dla kompletności, oto kod, który właśnie podałem (uwaga: bez sprawdzania błędów!), Który konfiguruje Unixowy TTY i emuluje <conio.h>
funkcje DOS kbhit()
i getch()
:
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/select.h>
#include <termios.h>
struct termios orig_termios;
void reset_terminal_mode()
{
tcsetattr(0, TCSANOW, &orig_termios);
}
void set_conio_terminal_mode()
{
struct termios new_termios;
tcgetattr(0, &orig_termios);
memcpy(&new_termios, &orig_termios, sizeof(new_termios));
atexit(reset_terminal_mode);
cfmakeraw(&new_termios);
tcsetattr(0, TCSANOW, &new_termios);
}
int kbhit()
{
struct timeval tv = { 0L, 0L };
fd_set fds;
FD_ZERO(&fds);
FD_SET(0, &fds);
return select(1, &fds, NULL, NULL, &tv);
}
int getch()
{
int r;
unsigned char c;
if ((r = read(0, &c, sizeof(c))) < 0) {
return r;
} else {
return c;
}
}
int main(int argc, char *argv[])
{
set_conio_terminal_mode();
while (!kbhit()) {
}
(void)getch();
}