C非阻塞键盘输入

时间:2009-01-15 23:28:34

标签: c linux asynchronous input nonblocking

我正在尝试用C编写一个程序(在Linux上)循环,直到用户按下一个键,但不应该要求按键继续每个循环。

有一种简单的方法吗?我想我可以用select()做到这一点,但这似乎很多工作。

或者,有没有办法在程序关闭之前捕获 ctrl - c 按键进行清理而不是非阻塞io?

10 个答案:

答案 0 :(得分:60)

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

但是请注意,对于后一种方法,您还需要设置TTY,使其一次一个字符而不是一次一行模式。后者是默认值 - 如果您输入一行文本,则在按Enter键之前不会将其发送到正在运行的程序的stdin。

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

为了完整起见,这里有一些代码我刚刚敲了(nb:没有错误检查!),它设置了一个Unix TTY并模拟了DOS <conio.h>函数kbhit()和{{1} }:

getch()

答案 1 :(得分:15)

为方便起见,

select()有点太低级了。我建议您使用ncurses库将终端设置为cbreak模式和延迟模式,然后调用getch(),如果没有准备就绪,将返回ERR

WINDOW *w = initscr();
cbreak();
nodelay(w, TRUE);

此时,您可以不加阻塞地致电getch

答案 2 :(得分:11)

在UNIX系统上,您可以使用sigaction调用来注册代表Control + C键序列的SIGINT信号的信号处理程序。信号处理程序可以设置一个标志,该标志将在循环中进行检查,使其适当中断。

答案 3 :(得分:6)

您可能需要kbhit();

//Example will loop until a key is pressed
#include <conio.h>
#include <iostream>

using namespace std;

int main()
{
    while(1)
    {
        if(kbhit())
        {
            break;
        }
    }
}

这可能不适用于所有环境。一种可移植的方法是创建监视线程并在getch();

上设置一些标记

答案 4 :(得分:3)

curses库可用于此目的。当然,select()和信号处理程序也可以在一定程度上使用。

答案 5 :(得分:3)

获取非阻塞键盘输入的另一种方法是打开设备文件并阅读它!

你必须知道你正在寻找的设备文件,/ dev / input / event *之一。您可以运行cat / proc / bus / input / devices来查找所需的设备。

此代码适用于我(以管理员身份运行)。

  #include <stdlib.h>
  #include <stdio.h>
  #include <unistd.h>
  #include <fcntl.h>
  #include <errno.h>
  #include <linux/input.h>

  int main(int argc, char** argv)
  {
      int fd, bytes;
      struct input_event data;

      const char *pDevice = "/dev/input/event2";

      // Open Keyboard
      fd = open(pDevice, O_RDONLY | O_NONBLOCK);
      if(fd == -1)
      {
          printf("ERROR Opening %s\n", pDevice);
          return -1;
      }

      while(1)
      {
          // Read Keyboard Data
          bytes = read(fd, &data, sizeof(data));
          if(bytes > 0)
          {
              printf("Keypress value=%x, type=%x, code=%x\n", data.value, data.type, data.code);
          }
          else
          {
              // Nothing read
              sleep(1);
          }
      }

      return 0;
   }

答案 6 :(得分:2)

如果你很高兴只是抓住Control-C,那就完成了。如果你真的想要非阻塞I / O但你不想要curses库,另一种方法是将lock,stock和barrel移动到AT&T sfio library。这是一个很好的库,在C stdio上有图案,但更灵活,线程安全,性能更好。 (sfio代表安全,快速的I / O.)

答案 7 :(得分:1)

没有可移植的方法来做到这一点,但select()可能是一个好方法。有关更多可能的解决方案,请参阅http://c-faq.com/osdep/readavail.html

答案 8 :(得分:0)

这是为您执行此操作的功能。您需要POSIX系统附带的termios.h

#include <termios.h>
void stdin_set(int cmd)
{
    struct termios t;
    tcgetattr(1,&t);
    switch (cmd) {
    case 1:
            t.c_lflag &= ~ICANON;
            break;
    default:
            t.c_lflag |= ICANON;
            break;
    }
    tcsetattr(1,0,&t);
}

打破这种情况:tcgetattr获取当前终端信息并将其存储在t中。如果cmd为1,则t中的本地输入标志将设置为非阻塞输入。否则重置。然后tcsetattr将标准输入更改为t

如果您未在程序结束时重置标准输入,则shell中会出现问题。

答案 9 :(得分:-1)

您可以使用以下选项执行此操作:

  int nfds = 0;
  fd_set readfds;
  FD_ZERO(&readfds);
  FD_SET(0, &readfds); /* set the stdin in the set of file descriptors to be selected */
  while(1)
  {
     /* Do what you want */
     int count = select(nfds, &readfds, NULL, NULL, NULL);
     if (count > 0) {
      if (FD_ISSET(0, &readfds)) {
          /* If a character was pressed then we get it and exit */
          getchar();
          break;
      }
     }
  }

没有太多工作:D