如果需要太长时间才能完成,则超时功能[重复]
- 2024-12-13 08:36:00
- admin 原创
- 113
问题描述:
我有一个 shell 脚本,它循环遍历包含我想要访问并截取屏幕截图的 URL 的文本文件。
这一切都完成了,而且很简单。脚本初始化一个类,该类在运行时会创建列表中每个站点的屏幕截图。有些站点需要很长时间才能加载,有些站点可能根本无法加载。所以我想将屏幕抓取函数包装在一个超时脚本中,False
如果函数无法在 10 秒内完成,则让它返回。
我对最简单的解决方案感到满意,也许设置一个异步计时器,无论函数内部实际发生什么,它都会在 10 秒后返回 False?
解决方案 1:
信号的文档中描述了操作超时的过程。
基本思想是使用信号处理程序在某个时间间隔内设置警报,并在计时器到期后引发异常。
请注意,这仅适用于 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():
...
解决方案 2:
我使用该声明重写了大卫的答案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)
相关推荐
热门文章
项目管理软件有哪些?
热门标签
云禅道AD