首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >在Ruby中检测不带getc/getc的按键(非阻塞)

在Ruby中检测不带getc/getc的按键(非阻塞)
EN

Stack Overflow用户
提问于 2009-06-03 19:39:30
回答 5查看 22.1K关注 0票数 20

我有一个简单的任务,需要等待文件系统上的变化(它本质上是一个原型编译器)。所以我有一个简单的无限循环,在检查更改的文件后有5秒的休眠。

代码语言:javascript
复制
loop do
  # if files changed
  #   process files
  #   and puts result
  sleep 5
end

Ctrl+C敬礼相比,我更希望能够测试并查看某个键是否已被按下,而不会阻塞循环。从本质上讲,我只需要一种方法来判断是否有传入的按键,然后一种方法来获取它们,直到满足Q,然后退出程序。

我想要的是:

代码语言:javascript
复制
def wait_for_Q
  key_is_pressed && get_ch == 'Q'
end

loop do
  # if files changed
  #   process files
  #   and puts result
  wait_for_Q or sleep 5
end

或者,这是Ruby做不好的事情吗?

EN

回答 5

Stack Overflow用户

发布于 2011-01-31 04:28:06

您也可以在没有缓冲区的情况下执行此操作。在基于unix的系统中,这很容易:

代码语言:javascript
复制
system("stty raw -echo") #=> Raw mode, no echo
char = STDIN.getc
system("stty -raw echo") #=> Reset terminal mode
puts char

这将等待一个键被按下并返回字符码。不需要施压。

char = STDIN.getc放入一个循环中,您就得到了它!

如果你是在windows上,根据Ruby的方式,你需要用C语言写一个扩展,或者使用这个小技巧(尽管这是在2001年写的,所以可能有更好的方法)

代码语言:javascript
复制
require 'Win32API'
char = Win32API.new('crtdll','_getch', [], 'L').Call

下面是我的参考资料:great book, if you don't own it you should

票数 10
EN

Stack Overflow用户

发布于 2012-11-07 02:30:40

其他答案的组合会得到所需的行为。在OSX和Linux上的ruby 1.9.3中测试。

代码语言:javascript
复制
loop do
  puts 'foo'
  system("stty raw -echo")
  char = STDIN.read_nonblock(1) rescue nil
  system("stty -raw echo")
  break if /q/i =~ char
  sleep(2)
end
票数 9
EN

Stack Overflow用户

发布于 2014-03-26 19:42:42

通过结合我刚刚读到的各种解决方案,我想出了一种跨平台的方法来解决这个问题。Details here,但下面是相关的代码:一个返回ASCII码的GetKey.getkey方法,如果没有按下任何代码,则返回nil

应该可以在Windows和Unix上运行。

代码语言:javascript
复制
module GetKey

  # Check if Win32API is accessible or not
  @use_stty = begin
    require 'Win32API'
    false
  rescue LoadError
    # Use Unix way
    true
  end

  # Return the ASCII code last key pressed, or nil if none
  #
  # Return::
  # * _Integer_: ASCII code of the last key pressed, or nil if none
  def self.getkey
    if @use_stty
      system('stty raw -echo') # => Raw mode, no echo
      char = (STDIN.read_nonblock(1).ord rescue nil)
      system('stty -raw echo') # => Reset terminal mode
      return char
    else
      return Win32API.new('crtdll', '_kbhit', [ ], 'I').Call.zero? ? nil : Win32API.new('crtdll', '_getch', [ ], 'L').Call
    end
  end

end

下面是一个简单的程序来测试它:

代码语言:javascript
复制
loop do
  k = GetKey.getkey
  puts "Key pressed: #{k.inspect}"
  sleep 1
end

在上面提供的链接中,我还展示了如何使用curses库,但结果在Windows上有点奇怪。

票数 7
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/946738

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档