超时功能,如果需要太长的时间才能完成

我有一个 shell 脚本,它循环遍历一个包含 URL: s 的文本文件,我想访问这个文件并截屏。

这一切都很简单。该脚本初始化一个类,该类在运行时创建列表中每个站点的屏幕快照。有些站点需要很长很长的时间才能加载,有些站点可能根本无法加载。因此,我想把屏幕截图器函数包装在一个超时脚本中,如果函数不能在10秒内完成,则返回 False

我满足于最简单的解决方案,也许设置一个异步计时器,不管函数内部实际发生了什么,10秒后返回 False?

222675 次浏览

信号的文档中描述了计时操作的过程。

其基本思想是使用信号处理程序设置一段时间间隔的警报,并在该计时器过期时引发异常。

注意,这只能在 UNIX 上使用。

下面是一个创建装饰器的实现(将下面的代码保存为 timeout.py)。

import errno
import os
import signal
import functools


class TimeoutError(Exception):
pass


def timeout(seconds=10, error_message=os.strerror(errno.ETIME)):
def decorator(func):
def _handle_timeout(signum, frame):
raise TimeoutError(error_message)


@functools.wraps(func)
def wrapper(*args, **kwargs):
signal.signal(signal.SIGALRM, _handle_timeout)
signal.alarm(seconds)
try:
result = func(*args, **kwargs)
finally:
signal.alarm(0)
return result


return wrapper


return decorator

这将创建一个名为 @timeout的装饰符,可以应用于任何长时间运行的函数。

因此,在应用程序代码中,可以这样使用装饰器:

from timeout import timeout


# Timeout a long running function with the default expiry of 10 seconds.
@timeout
def long_running_function1():
...


# Timeout after 5 seconds
@timeout(5)
def long_running_function2():
...


# Timeout after 30 seconds, with the error "Connection timed out"
@timeout(30, os.strerror(errno.ETIMEDOUT))
def long_running_function3():
...

我用 with语句重写了大卫的答案,它允许你这样做:

with timeout(seconds=3):
time.sleep(4)

它将引发 TimeoutError。

代码仍然使用 signal,因此只能使用 UNIX:

import signal


class timeout:
def __init__(self, seconds=1, error_message='Timeout'):
self.seconds = seconds
self.error_message = error_message
def handle_timeout(self, signum, frame):
raise TimeoutError(self.error_message)
def __enter__(self):
signal.signal(signal.SIGALRM, self.handle_timeout)
signal.alarm(self.seconds)
def __exit__(self, type, value, traceback):
signal.alarm(0)