」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 如何檢測從終端運行的 Python 腳本中的鍵盤輸入?

如何檢測從終端運行的 Python 腳本中的鍵盤輸入?

發佈於2024-12-21
瀏覽:112

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)
最新教學 更多>
  • 儘管程式碼有效,為什麼 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 月 23 日 ...
    程式設計 發佈於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 連線的缺點:意外後果雖然PDO 中的持久連線旨在透過快取和重複使用連線來增強效能,但它們可以還會引入可能影響效能的意外後果。 事務和連接狀態問題:持久連接的一個顯著缺點是意外的腳本終止會留下打開的連接,這可能會導致各種問題:鎖定表: 如果死腳本鎖定...
    程式設計 發佈於2024-12-21
  • 大批
    大批
    方法是可以在物件上呼叫的 fns 數組是對象,因此它們在 JS 中也有方法。 slice(begin):將陣列的一部分提取到新數組中,而不改變原始數組。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index ...
    程式設計 發佈於2024-12-21
  • 如何在 PHP 中顯示 MySQL BLOB 映像?
    如何在 PHP 中顯示 MySQL BLOB 映像?
    從MySQL 顯示PHP BLOB 影像背景在MySQL 資料庫中將影像作為二進位大物件(BLOB) 儲存和擷取是一種常見技術。然而,顯示這些圖像有時可能具有挑戰性。 解決方案解決方案插入映像:使用file_get_contents 函數讀取映像$query = "INSERT INTO ...
    程式設計 發佈於2024-12-21
  • 在 Go 中使用 WebSocket 進行即時通信
    在 Go 中使用 WebSocket 進行即時通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSocke...
    程式設計 發佈於2024-12-21
  • 深入研究 Reactjs
    深入研究 Reactjs
    模因生成器 Meme Generator 是一個有趣的互動式 Web 應用程序,允許用戶使用各種模板創建自訂 Meme。我已經透過教育平台學習軟體開發幾個月了,這個專案是課程的一部分。 目錄 示範 特徵 技術堆疊 安裝 用法 作者 執照 表達您的支持 ...
    程式設計 發佈於2024-12-21
  • 如何使用 CSS 禁用連結?
    如何使用 CSS 禁用連結?
    CSS停用連結技巧問題:是否可以使用CSS停用連結?例如,如果您有一個名為“當前頁面”的類,您可能希望阻止該類的連結在單擊時處於活動狀態。 解決方案:以下 CSS 程式碼片段提供了一個簡單的解決方案:[aria-current="page"] { pointer-events...
    程式設計 發佈於2024-12-21
  • 如何讓我的 Python 類別 JSON 可序列化?
    如何讓我的 Python 類別 JSON 可序列化?
    使 Python 類別 JSON 可序列化序列化允許將物件轉換為適合持久或傳輸的格式。在 JSON 中序列化自訂類別通常會導致 TypeError,因為它們的非 JSON 可序列化性質。 實作 toJSON() 方法一種方法是建立一個類別的 toJSON() 方法,手動處理其序列化。這需要實作自訂 ...
    程式設計 發佈於2024-12-21
  • 如何在 PHP 中正確使用 INSERT INTO 準備語句?
    如何在 PHP 中正確使用 INSERT INTO 準備語句?
    將準備好的語句與INSERT INTO 合併在遍歷PHP 的迷宮深度:資料物件時,在嘗試執行MySQL 時出現了一個令人困惑的難題使用準備好的語句進行查詢,特別是用於INSERT INTO 操作。考慮以下程式碼片段:$statement = $link->prepare("INSER...
    程式設計 發佈於2024-12-21
  • 為什麼 `Class object();` 不建立 C++ 物件?
    為什麼 `Class object();` 不建立 C++ 物件?
    瞭解建構函式呼叫的缺失在 C 中,建構子用來初始化類別的物件。然而,在提供的程式碼片段中,構造函數並沒有被調用,導致意外的行為。 問題有問題的行:Class object();其實並沒有呼叫 Class 類別的建構子。相反,它宣告一個傳回 Class 物件的函數。要正確呼叫建構函式並建立對象,必須在...
    程式設計 發佈於2024-12-21

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3