BLOG
Enjoy when you can, and endure when you must.
OCT 15, 2014/Python
Python实现按任意键继续

上班时间,好基友向我提了个问题:在 Python 中如何做到按任意键退出?第一反应就是这曾经实现过!不过第二反应...忘了,赶紧补补。

要实现该功能,需要的就是暂停程序、等待并捕捉用户的一个键盘输入,然后继续执行。Python 有内建的库能帮我们实现该功能,不过要区别对待 Windows 和 Linux。

msvcrt 中的 getch() 方法 能够帮助在 Windows 下实现,其作用是获取一个按键响应并返回对应的字符。它并不在命令行中回显。有如下程序段:

import msvcrt
print ord(msvcrt.getch())

这里利用 ord 将获得的字符转换为 ASCII 数值,例如捕获按键“d”(注意是小写)将得到数值100。

Linux 下呢?嗯,相对复杂一点点,不过先理清思路的话就好办了。首先要知晓一下 Linux 终端的三种模式,分别为规范模式、非规范模式和 raw 模式:

规范模式

规范模式,也被成为 cooked 模式,是用户常见的模式。驱动程序输入的字符保存在缓冲区,并且仅在接收到回车键时才将这些缓冲的字符发送到程序。缓冲数据使驱动程序可以实现最基本的编辑功能, 被指派这些功能的特定键在驱动程序里设置,可以通过命令 stty 或系统调用 tcsetattr 来修改。

非规范模式

当缓冲和编辑功能被关闭时,连接被成为非规范模式。终端处理器仍旧进行特定的字符处理,例如处理 Ctrl-C 及换行符之间的转换,但是编辑键将没有意义,因此相应的输入被视为常规的数据输入,程序需要自己实现编辑功能。

raw 模式

当所有处理都被关闭后,驱动程序将输入直接传递给程序,连接被成为 raw 模式。

这里我们需要借助于非规范模式,那么要实现刚才 Windows 上的相似行为,需要以下代码:

import os
import termios

# 获取标准输入的描述符
fd = sys.stdin.fileno()

# 获取标准输入(终端)的设置
old_ttyinfo = termios.tcgetattr(fd)

# 配置终端
new_ttyinfo = old_ttyinfo[:]

# 使用非规范模式(索引3是c_lflag 也就是本地模式)
new_ttyinfo[3] &= ~termios.ICANON

# 关闭回显(输入不会被显示)
new_ttyinfo[3] &= ~termios.ECHO

# 使设置生效
termios.tcsetattr(fd, termios.TCSANOW, new_ttyinfo)

# 从终端读取
print ord(os.read(fd, 7))

由此看来,我们只需利用上面的方法捕获一个按键响应,然后继续程序即可做到按任意键继续或退出的功能了。当然,做到按指定键继续或退出的功能也可以按类似的方法实现,例如:

import msvcrt

print("Press 'D' to exit...")

while True:
    if ord(msvcrt.getch()) in [68, 100]:
        break

这样,当用户按下“D”或“d”时,则程序退出。

参考资料(本文部分转载自下列文章):

Linux 下 Python 实现按任意键退出

COMMENTS
22/07From aa

命令无效,不是等待输入,而是直接显示255

07/11From QM

请问怎么改回去呢

06/11From Danny

@刘国兵: 不是的哦,自己写的,基于 Django Web FrameWork。

29/10From 刘国兵

您好,请问你的博客是用啥搭建的。wordpress吗?

26/05From VS

很喜欢你的博客。

26/11From 啊啊啊

真不错

LEAVE COMMNT