如果完成时间太长,则超时功能[重复]

Posted

技术标签:

【中文标题】如果完成时间太长,则超时功能[重复]【英文标题】:Timeout function if it takes too long to finish [duplicate] 【发布时间】:2011-01-17 22:04:03 【问题描述】:

我有一个 shell 脚本,它遍历一个包含我想要访问的 URL:s 的文本文件并对其进行截图。

所有这些都已完成并且很简单。该脚本初始化一个类,该类在运行时会创建列表中每个站点的屏幕截图。有些网站需要很长时间才能加载,而有些网站可能根本没有加载。所以我想将screengrabber函数包装在一个超时脚本中,如果它不能在10秒内完成,则使函数返回False

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

【问题讨论】:

对于所有喜欢使用库而不是从 *** 复制+粘贴代码 sn-ps 的懒人:pypi.python.org/pypi/timeout-decorator 【参考方案1】:

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

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

请注意,这只适用于 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():
    ...

【讨论】:

请注意这不是线程安全的:如果您使用多线程,信号将被随机线程捕获。不过对于单线程程序,这是最简单的解决方案。 不错。另外,建议用@functools.wraps(func)装饰函数wrapper 仅供参考,第一个“@timeout”之后缺少括号。它应该是@timeout() def ... @wim 我认为它只能在主线程中使用,因为如果你在工作线程中使用它,它会引发'ValueError:信号仅在主线程中工作'。 在使用线程时,什么是可行的替代方案?【参考方案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)

【讨论】:

Python ***.com/a/1319675/380038 你可以很容易地添加一个装饰器@timeout.timeout作为一个静态方法。然后,您可以轻松地在装饰器或 with 语句之间进行选择。 有趣的是,如果在 with Timeout(t) 上下文中引发任何错误,__exit__ 仍会被调用,从而避免因引发 TimeOutError 而不是真正的错误而导致的任何并发症.这是一个非常可爱的解决方案。 有人可以推荐一个可行的解决方案,像这样,在线程中工作吗? @Nick 前段时间我创建了超时装饰器的版本,它适用于浮点数 - ***.com/questions/11901328/…

以上是关于如果完成时间太长,则超时功能[重复]的主要内容,如果未能解决你的问题,请参考以下文章

需要一些帮助来完成这个功能[重复]

等待先前的异步功能完成[重复]

Future和Promise

bash只读取用户输入的第一个字符的超时

AngularJS取消超时

如何为 Task 添加超时功能