”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 如何检测从终端运行的 Python 脚本中的键盘输入?

如何检测从终端运行的 Python 脚本中的键盘输入?

发布于2024-12-21
浏览:644

How do I detect keyboard input in a Python script running from the terminal?

如何从终端检测脚本中的键盘输入?

同步/阻塞按键捕获:

简单输入或 raw_input,阻塞函数,一旦用户按下换行符,就会返回用户输入的文本。

typedString = raw_input()

一个简单的阻塞函数,等待用户按下单个键,然后返回该键

class _Getch:
 """Gets a single character from standard input.  Does not echo to the
screen. From http://code.activestate.com/recipes/134892/"""
def __init__(self):
  try:
    self.impl = _GetchWindows()
  except ImportError:
    try:
      self.impl = _GetchMacCarbon()
    except(AttributeError, ImportError):
      self.impl = _GetchUnix()

def __call__(self): return self.impl()


class _GetchUnix:
def __init__(self):
  import tty, sys, termios # import termios now or else you'll get the Unix version on the Mac

def __call__(self):
  import sys, tty, termios
  fd = sys.stdin.fileno()
  old_settings = termios.tcgetattr(fd)
  try:
    tty.setraw(sys.stdin.fileno())
    ch = sys.stdin.read(1)
  finally:
    termios.tcsetattr(fd, termios.TCSADRAIN, old_settings)
  return ch

class _GetchWindows:
def __init__(self):
  import msvcrt

def __call__(self):
  import msvcrt
  return msvcrt.getch()

class _GetchMacCarbon:
"""
A function which returns the current ASCII key that is down;
if no ASCII key is down, the null string is returned.  The
page http://www.mactech.com/macintosh-c/chap02-1.html was
very helpful in figuring out how to do this.
"""
def __init__(self):
  import Carbon
  Carbon.Evt #see if it has this (in Unix, it doesn't)

def __call__(self):
  import Carbon
  if Carbon.Evt.EventAvail(0x0008)[0]==0: # 0x0008 is the keyDownMask
    return ''
  else:
    #
    # The event contains the following info:
    # (what,msg,when,where,mod)=Carbon.Evt.GetNextEvent(0x0008)[1]
    #
    # The message (msg) contains the ASCII char which is
    # extracted with the 0x000000FF charCodeMask; this
    # number is converted to an ASCII character with chr() and
    # returned
    #
    (what,msg,when,where,mod)=Carbon.Evt.GetNextEvent(0x0008)[1]
    return chr(msg & 0x000000FF)


def getKey():
  inkey = _Getch()
  import sys
  for i in xrange(sys.maxint):
    k=inkey()
    if k<>'':break

  return k

异步按键捕获:

每当用户在命令提示符中键入按键时,即使在解释器(键盘记录器)中键入内容,也会通过按下的按键调用回调

回调尚未实现。

用户按 Enter 键后使用键入的文本调用的回调(a较少的实时键盘记录器)

回调尚未实现。

程序运行时(例如,在 for 循环中)按下按键时调用的回调或 while 循环)

Windows:

import threading
from win32api import STD_INPUT_HANDLE
from win32console import GetStdHandle, KEY_EVENT, ENABLE_ECHO_INPUT, ENABLE_LINE_INPUT, ENABLE_PROCESSED_INPUT


class KeyAsyncReader():
  def __init__(self):
    self.stopLock = threading.Lock()
    self.stopped = True
    self.capturedChars = ""

    self.readHandle = GetStdHandle(STD_INPUT_HANDLE)
    self.readHandle.SetConsoleMode(ENABLE_LINE_INPUT|ENABLE_ECHO_INPUT|ENABLE_PROCESSED_INPUT)



  def startReading(self, readCallback):
    self.stopLock.acquire()

    try:
      if not self.stopped:
        raise Exception("Capture is already going")

      self.stopped = False
      self.readCallback = readCallback

      backgroundCaptureThread = threading.Thread(target=self.backgroundThreadReading)
      backgroundCaptureThread.daemon = True
      backgroundCaptureThread.start()
    except:
      self.stopLock.release()
      raise

    self.stopLock.release()


  def backgroundThreadReading(self):
    curEventLength = 0
    curKeysLength = 0
    while True:
      eventsPeek = self.readHandle.PeekConsoleInput(10000)

      self.stopLock.acquire()
      if self.stopped:
        self.stopLock.release()
        return
      self.stopLock.release()


      if len(eventsPeek) == 0:
        continue

      if not len(eventsPeek) == curEventLength:
        if self.getCharsFromEvents(eventsPeek[curEventLength:]):
          self.stopLock.acquire()
          self.stopped = True
          self.stopLock.release()
          break

        curEventLength = len(eventsPeek)



  def getCharsFromEvents(self, eventsPeek):
    callbackReturnedTrue = False
    for curEvent in eventsPeek:
      if curEvent.EventType == KEY_EVENT:
              if ord(curEvent.Char) == 0 or not curEvent.KeyDown:
                pass
              else:
                curChar = str(curEvent.Char)
                if self.readCallback(curChar) == True:
                  callbackReturnedTrue = True


    return callbackReturnedTrue

  def stopReading(self):
    self.stopLock.acquire()
    self.stopped = True
    self.stopLock.release()

轮询:

用户只是希望能够在按下某个键时执行某些操作,而不必等待该键(因此这应该是非-阻塞)。因此,他们调用 poll() 函数,该函数要么返回一个键,要么返回 None。这可以是有损的(如果它们在轮询之间花费太长时间,它们可能会错过一个键)或非有损的(轮询器将存储所有按下的键的历史记录,因此当 poll() 函数请求它们时,它们将始终被返回按按下的顺序)。

Windows 和 OS X(也许还有 Linux):

global isWindows

isWindows = False
try:
  from win32api import STD_INPUT_HANDLE
  from win32console import GetStdHandle, KEY_EVENT, ENABLE_ECHO_INPUT, ENABLE_LINE_INPUT, ENABLE_PROCESSED_INPUT
  isWindows = True
except ImportError as e:
  import sys
  import select
  import termios


class KeyPoller():
  def __enter__(self):
    global isWindows
    if isWindows:
      self.readHandle = GetStdHandle(STD_INPUT_HANDLE)
      self.readHandle.SetConsoleMode(ENABLE_LINE_INPUT|ENABLE_ECHO_INPUT|ENABLE_PROCESSED_INPUT)
      
      self.curEventLength = 0
      self.curKeysLength = 0
      
      self.capturedChars = []
    else:
      # Save the terminal settings
      self.fd = sys.stdin.fileno()
      self.new_term = termios.tcgetattr(self.fd)
      self.old_term = termios.tcgetattr(self.fd)
      
      # New terminal setting unbuffered
      self.new_term[3] = (self.new_term[3] & ~termios.ICANON & ~termios.ECHO)
      termios.tcsetattr(self.fd, termios.TCSAFLUSH, self.new_term)
      
    return self
  
  def __exit__(self, type, value, traceback):
    if isWindows:
      pass
    else:
      termios.tcsetattr(self.fd, termios.TCSAFLUSH, self.old_term)
  
  def poll(self):
    if isWindows:
      if not len(self.capturedChars) == 0:
        return self.capturedChars.pop(0)

      eventsPeek = self.readHandle.PeekConsoleInput(10000)
最新教程 更多>
  • 如何在 C++ 中定义静态 const std::string 成员?
    如何在 C++ 中定义静态 const std::string 成员?
    定义 const std::string 类型的静态数据成员在 C 中,定义 std::string 类型的私有静态 const 成员在类内使用类内初始化,如下所示,不符合C标准:class A { private: static const string RECTANGLE = ...
    编程 发布于2024-12-21
  • 使用 Uvicorn 在 FastAPI 中发出并发 HTTP 请求时如何避免“ConnectionClosed”错误?
    使用 Uvicorn 在 FastAPI 中发出并发 HTTP 请求时如何避免“ConnectionClosed”错误?
    在 Uvicorn/FastAPI 中发出 HTTP 请求处理使用 FastAPI 和 Uvicorn 构建的 HTTP 端点时,通常会从外部 API 请求数据。但是,在处理多个并发请求时,可能会出现“can't handle event type ConnectionClosed when...
    编程 发布于2024-12-21
  • 如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    如何修复 macOS 上 Django 中的“配置不正确:加载 MySQLdb 模块时出错”?
    MySQL配置不正确:相对路径的问题在Django中运行python manage.py runserver时,可能会遇到以下错误:ImproperlyConfigured: Error loading MySQLdb module: dlopen(/Library/Python/2.7/site-...
    编程 发布于2024-12-21
  • 如何使用非标准证书文件在Go Web服务器上建立HTTPS?
    如何使用非标准证书文件在Go Web服务器上建立HTTPS?
    如何使用非标准证书文件在 Go Web 服务器上建立 HTTPS提供的文档建议连接三个 .pem 文件。但是,如果您没有这些文件,以下是如何使用您拥有的证书文件设置 HTTPS:组合中间证书:虽然 Go 通常需要一个串联的证书文件,其他平台仅存储根证书。为了确保兼容性,请连接中间证书:cat web...
    编程 发布于2024-12-21
  • 如何可靠地处理带有子元素的绝对定位 div 上的鼠标移出事件?
    如何可靠地处理带有子元素的绝对定位 div 上的鼠标移出事件?
    在没有 jQuery 的情况下处理带有子元素的绝对 Div 中的 Mouseout 事件处理绝对定位的 div 时,处理 mouseout 事件可能具有挑战性。默认情况下,如果鼠标悬停在父 div 内的子元素上,则在鼠标退出外部 div 之前,mouseout 事件会提前触发。要解决此问题,请考虑使...
    编程 发布于2024-12-21
  • PHP 的 `==` 和 `===` 运算符有什么区别?
    PHP 的 `==` 和 `===` 运算符有什么区别?
    PHP Double (==) 和 Triple (===) 相等比较有何不同?在 PHP 中比较值时,两个可以使用不同的运算符:松散相等 (==) 运算符和严格相同 (===) 运算符。了解它们的细微差别对于确保可靠的比较至关重要。松散相等 (==) 比较松散相等运算符在比较值之前执行类型杂乱操作...
    编程 发布于2024-12-21
  • 如何在 JavaScript 中对字符串执行数学运算?
    如何在 JavaScript 中对字符串执行数学运算?
    将字符串转换为数字进行数学运算尽管包含数字字符,但字符串不能直接作为数字进行操作。要执行算术运算,必须首先将它们转换为数字形式。考虑给出的示例:var num1 = '20', num2 = '30.5';直接添加这些字符串会导致串联:num1 num2; // '2030.5'要强制将...
    编程 发布于2024-12-21
  • 尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    尽管代码有效,为什么 POST 请求无法捕获 PHP 中的输入?
    解决 PHP 中的 POST 请求故障在提供的代码片段中:action=''而不是:action="<?php echo $_SERVER['PHP_SELF'];?>";?>"检查 $_POST数组:表单提交后使用 var_dump 检查 $_POST 数...
    编程 发布于2024-12-21
  • 为什么我的 Div 重叠?了解并解决保证金崩溃问题
    为什么我的 Div 重叠?了解并解决保证金崩溃问题
    边距折叠:理解和解决 Div 布局中的重叠设计具有多个 div 元素的布局时,理解边距的概念至关重要折叠以避免不必要的重叠边距。边距折叠是一种 CSS 行为,当相邻元素的边距合并在一起时会发生,有效地增加了它们之间的总边距空间。边距折叠的原因在您的具体情况,重叠边距可能是由于以下 CSS 规则的组合...
    编程 发布于2024-12-21
  • 第一个 PHP 8.3 候选版本现已可供测试
    第一个 PHP 8.3 候选版本现已可供测试
    即将推出的 PHP 版本的第一个候选版本 PHP 8.3 现已可供测试。 第一个普遍可用的 PHP 版本计划于今年 11 月 23 日发布,PHP 8.3 的第一个候选版本是一个重要的预发布里程碑,因为它表明了 PHP 8.3 的所有更改现在已经实现,只剩下完善和错误修复了。 在 11 月 2...
    编程 发布于2024-12-21
  • 插入数据时如何修复“常规错误:2006 MySQL 服务器已消失”?
    插入数据时如何修复“常规错误:2006 MySQL 服务器已消失”?
    插入记录时如何解决“一般错误:2006 MySQL 服务器已消失”介绍:将数据插入 MySQL 数据库有时会导致错误“一般错误:2006 MySQL 服务器已消失”。当与服务器的连接丢失时会出现此错误,通常是由于 MySQL 配置中的两个变量之一所致。解决方案:解决此错误的关键是调整wait_tim...
    编程 发布于2024-12-21
  • 如何正确处理 JSON 数据中的换行符?
    如何正确处理 JSON 数据中的换行符?
    处理 JSON 中的换行符处理 JSON 数据时,必须正确处理换行符以避免意外错误。下面是该问题及其解决方案的详细说明。问题使用 eval 或 JSON.parse 解析包含换行符的 JSON 数据时,可能会遇到这样的错误作为“未终止的字符串文字”。这是因为 JSON 中的双引号字符串中无法识别换行...
    编程 发布于2024-12-21
  • Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta 中的列偏移发生了什么?
    Bootstrap 4 Beta:列偏移的删除和恢复Bootstrap 4 在其 Beta 1 版本中引入了重大更改柱子偏移了。然而,随着 Beta 2 的后续发布,这些变化已经逆转。从 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    编程 发布于2024-12-21
  • 您是否应该使用持久 PDO 连接:权衡性能收益与潜在风险?
    您是否应该使用持久 PDO 连接:权衡性能收益与潜在风险?
    使用持久 PDO 连接的缺点:意外后果虽然 PDO 中的持久连接旨在通过缓存和重用连接来增强性能,但它们可以还会引入可能影响性能的意外后果。事务和连接状态问题:持久连接的一个显着缺点是意外的脚本终止会留下打开的连接,这可能会导致各种问题:锁定表: 如果死脚本锁定了表,它们将保持锁定状态,直到持久连接...
    编程 发布于2024-12-21
  • 大批
    大批
    方法是可以在对象上调用的 fns 数组是对象,因此它们在 JS 中也有方法。 slice(begin):将数组的一部分提取到新数组中,而不改变原始数组。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index p...
    编程 发布于2024-12-21

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3