小编典典

Python-超时功能,如果完成时间太长

python

我有一个Shell脚本,该脚本遍历包含要访问的URL:s并截取其屏幕截图的文本文件。

所有这一切都完成且简单。该脚本会初始化一个类,该类在运行时会创建列表中每个站点的屏幕快照。某些站点的加载时间非常非常长,而某些站点可能根本无法加载。因此,我想将screengrabber函数包装在一个超时脚本中,以使该函数False在10秒内无法完成时返回。

我对最简单的解决方案感到满意,也许设置一个异步计时器,无论函数内部实际发生什么,该计时器都会在10秒后返回False


阅读 1053

收藏
2020-02-14

共1个答案

小编典典

signal的文档中描述了用于使操作超时的过程。

基本思想是使用信号处理程序在某个时间间隔内设置警报,并在计时器到期后引发异常。

请注意,这仅适用于UNIX。

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

from functools import wraps
import errno
import os
import signal

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)

        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 wraps(func)(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():
    ...
2020-02-14