Linux非阻塞键盘监听,非阻塞键盘输入

如前所述,您可以使用sigaction捕捉ctrl-c,或select捕捉任何标准输入。

但是,请注意,对于后一种方法,您还需要设置TTY,使其处于实时字符模式,而不是逐时模式。后者是默认的-如果键入一行文本,它不会被发送到正在运行的程序的stdin,直到您按Enter键。

您需要使用tcsetattr()函数关闭ICANON模式,也可能也禁用回波。从内存中,您还必须将终端设置回ICANON模式,当程序退出!

为了完整起见,下面是我刚刚编写的一些代码(nb:没有错误检查!),它设置一个UnixTTY并模拟DOS功能kbhit()和getch():#include #include #include #include #include struct termios orig_termios;void reset_terminal_mode(){

tcsetattr(0, TCSANOW, &orig_termios);}void set_conio_terminal_mode(){

struct termios new_termios;

/* take two copies - one for now, one for later */

tcgetattr(0, &orig_termios);

memcpy(&new_termios, &orig_termios, sizeof(new_termios));

/* register cleanup handler, and set the new terminal mode */

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))) 

return r;

} else {

return c;

}}int main(int argc, char *argv[]){

set_conio_terminal_mode();

while (!kbhit()) {

/* do some work */

}

(void)getch(); /* consume the character */}

<< 键盘监听线程可以通过创建一个独立运行的线程来不断检测用户是否按下了某个特定按键,并根据按键执行相应的操作。以下是如何在 C 中使用 POSIX 线程(pthread)进行键盘监听并获取键值的一个示例。 需要注意的是,在标准 C 输入中直接捕获单个字符而不等待回车通常需要借助外部库或更改终端的行为设置(如关闭输入缓冲区)。下面的例子结合 `termios` 来修改 Linux 终端行为以实现非阻塞键盘读取。 ### 示例代码 ```c #include <stdio.h> #include <stdlib.h> #include <pthread.h> #include <unistd.h> #include <fcntl.h> #include <termios.h> // 保存原始 terminal 属性以便恢复 struct termios orig_termios; void disableRawMode() { tcsetattr(STDIN_FILENO, TCSAFLUSH, &orig_termios); } void enableRawMode() { struct termios raw; if (tcgetattr(STDIN_FILENO, &raw) == -1) { perror("tcgetattr"); exit(1); } atexit(disableRawMode); // 在程序退出时自动调用disableRawMode() raw.c_lflag &= ~(ICANON | ECHO); // 关闭 ICANON 和 ECHO raw.c_cc[VMIN] = 0; // 最小字节数为0 raw.c_cc[VTIME] = 1; // 设置超时时间,单位十分之一秒 if (tcsetattr(STDIN_FILENO, TCSAFLUSH, &raw) == -1) { perror("tcsetattr"); exit(1); } orig_termios = raw; // 保存原来的配置用于之后恢复 } // 键盘监听函数 void* keyboardListener(void *arg) { char ch; while (read(STDIN_FILENO, &ch, 1) == 1 && ch != 'q') { printf("Key pressed: %d ('%c')\n", ch, ch); } return NULL; } int main() { pthread_t tid; enableRawMode(); if(pthread_create(&tid, NULL, keyboardListener, NULL)) { fprintf(stderr,"Error creating thread\n"); return 1; } printf("Press keys or press 'q' to quit.\n"); void *status; if(pthread_join(tid,&status)) { fprintf(stderr,"Error joining thread\n"); return 2; } disableRawMode(); printf("\nExited Keyboard Listener Thread.\n"); return 0; } ``` **解释** 1. **enableRawMode**: 此函数用来将终端置于 "raw mode" 模式下。这种模式去掉了所有行编辑功能(例如退格、删除等),并且每个按下都会立刻返回给应用程序处理。 2. **keyboardListener 函数**: 这是一个单独的线程函数,它会持续从标准输入流中读取单个字符直到接收到 'q' 字符为止。 3. **main 函数**: 首先启用 Raw Mode 并启动一个新的线程来进行键盘监控。主线程则只负责初始化与清理任务;当子线程结束时,重新切换回到普通模式。 此代码仅适用于基于 Unix 的系统包括 macOS 及大多数 GNU/Linux 发行版上编译运行成功。对于 Windows 则需采用其他方法比如 _getch() 或 WinAPI 提供的相关接口完成类似效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值