Если допустимо блокировать основной поток, когда пользователь не предоставил ответа:
from threading import Timer
timeout = 10
t = Timer(timeout, print, ['Sorry, times up'])
t.start()
prompt = "You have %d seconds to choose the correct answer...\n" % timeout
answer = input(prompt)
t.cancel()
В противном случае вы могли бы использовать ответ @Alex Martelli (измененный для Python 3) в Windows (не тестировалось):
import msvcrt
import time
class TimeoutExpired(Exception):
pass
def input_with_timeout(prompt, timeout, timer=time.monotonic):
sys.stdout.write(prompt)
sys.stdout.flush()
endtime = timer() + timeout
result = []
while timer() < endtime:
if msvcrt.kbhit():
result.append(msvcrt.getwche()) #XXX can it block on multibyte characters?
if result[-1] == '\n': #XXX check what Windows returns here
return ''.join(result[:-1])
time.sleep(0.04) # just to yield to other processes/threads
raise TimeoutExpired
Использование:
try:
answer = input_with_timeout(prompt, 10)
except TimeoutExpired:
print('Sorry, times up')
else:
print('Got %r' % answer)
В Unix вы можете попробовать:
import select
import sys
def input_with_timeout(prompt, timeout):
sys.stdout.write(prompt)
sys.stdout.flush()
ready, _, _ = select.select([sys.stdin], [],[], timeout)
if ready:
return sys.stdin.readline().rstrip('\n') # expect stdin to be line-buffered
raise TimeoutExpired
Или:
import signal
def alarm_handler(signum, frame):
raise TimeoutExpired
def input_with_timeout(prompt, timeout):
# set signal handler
signal.signal(signal.SIGALRM, alarm_handler)
signal.alarm(timeout) # produce SIGALRM in `timeout` seconds
try:
return input(prompt)
finally:
signal.alarm(0) # cancel alarm
Я исследовал эту тему год назад, ища объектную модель для генерации RDL в памяти. Не было, но ходят слухи. Быстрое расследование в настоящее время: объектная модель под названием RDLOM существует, но на самом деле не поддерживается Microsoft .
В моем реальном подходе используется самогенерируемая объектная модель, построенная на схеме RDL .
Это далеко не идеальное решение, потому что вам нужно знать сгенерированный объект в деталях, а также потому, что код создания объекта действительно уродлив. Но другие обходные пути, которые я обнаружил в своем исследовании, используют XML или XSLT напрямую для генерации RDL, и они по-своему плохи.