我有一个无限while循环,当用户按键时,我想中断它。通常我使用raw_input
来获得用户的响应;但是,我需要raw_input
不要等待响应。我想要这样的东西:
print 'Press enter to continue.'
while True:
# Do stuff
#
# User pressed enter, break out of loop
这应该很简单,但我似乎想不通。我倾向于使用线程的解决方案,但我宁愿不必这么做。我怎样才能做到这一点?
您可以使用来自stdin:的非阻塞读取
import sys
import os
import fcntl
import time
fl = fcntl.fcntl(sys.stdin.fileno(), fcntl.F_GETFL)
fcntl.fcntl(sys.stdin.fileno(), fcntl.F_SETFL, fl | os.O_NONBLOCK)
while True:
print("Waiting for user input")
try:
stdin = sys.stdin.read()
if "n" in stdin or "r" in stdin:
break
except IOError:
pass
time.sleep(1)
我认为你可以用msvcrt:做得更好
import msvcrt, time
i = 0
while True:
i = i + 1
if msvcrt.kbhit():
if msvcrt.getwche() == 'r':
break
time.sleep(0.1)
print(i)
遗憾的是,仍然是特定窗口。
在python 3.5上,您可以使用以下代码。它可以针对特定的击键进行调整。while循环将一直运行,直到用户按键为止。
import time
import threading
# set global variable flag
flag = 1
def normal():
global flag
while flag==1:
print('normal stuff')
time.sleep(2)
if flag==False:
print('The while loop is now closing')
def get_input():
global flag
keystrk=input('Press a key n')
# thread doesn't continue until key is pressed
print('You pressed: ', keystrk)
flag=False
print('flag is now:', flag)
n=threading.Thread(target=normal)
i=threading.Thread(target=get_input)
n.start()
i.start()
我无法得到一些流行的答案。所以我想出了另一种方法,使用CTRL
+C
插入用户输入并吸收键盘中断。一个简单的解决方案可以是使用try-catch块,
i = 0
try:
while True:
i+=1
print(i)
sleep(1)
except:
pass
# do what you want to do after it...
我是从运行flask和django这样的服务器中得到这个想法的。这可能与OP的要求略有不同,但它可能会帮助其他想要类似东西的人。
使用msvcrt
模块作为主要推荐模块,我能够想出一些有效的方法。以下是一个基本示例,如果按下任何键,将退出循环,而不仅仅是输入。
import msvcrt, time
i = 0
while True:
i = i + 1
if msvcrt.kbhit():
break
time.sleep(0.1)
print i
您需要的是一个非阻塞的原始输入,如果您不想使用线程,下面有一个简单的解决方案,他会超时20毫秒,然后如果用户没有按键,则引发和异常,如果他按下了,则类返回按下的键。
import signal
class AlarmException(Exception):
pass
def alarmHandler(signum, frame):
raise AlarmException
def nonBlockingRawInput(prompt='', timeout=20):
signal.signal(signal.SIGALRM, alarmHandler)
signal.alarm(timeout)
try:
text = raw_input(prompt)
signal.alarm(0)
return text
except AlarmException:
print 'nPrompt timeout. Continuing...'
signal.signal(signal.SIGALRM, signal.SIG_IGN)
return ''
源代码
我定义了一个函数,该函数要求用户输入数字并返回该数字的阶乘。如果用户想停止,他们必须按0,然后它将退出循环。我们可以指定任何特定的键来输入"n"以退出循环。
import math
def factorial_func(n):
return math.factorial(n)
while True:
n = int(input("Please enter the number to find factorial: "))
print(factorial_func(n))
if n == 0:
exit()