1. ホーム
  2. python

[解決済み] 終了までに時間がかかりすぎる場合のタイムアウト機能【重複】について

2022-03-03 15:07:42

質問

私は、訪問してスクリーンショットを撮りたいURLを含むテキストファイルをループするシェルスクリプトを持っています。

これだけのことで、シンプルにできています。スクリプトはクラスを初期化し、それを実行するとリスト内の各サイトのスクリーンショットが作成されます。サイトによっては、読み込みに非常に長い時間がかかるものや、まったく読み込まれないものもある。そこで、スクリーングラバーファンクションをタイムアウトスクリプトでラップし、ファンクションが以下を返すようにしたい。 False 10秒以内に終了しなかった場合。

できるだけシンプルな解決策に満足しています。非同期タイマーを設定して、関数内で実際に何が起こっても、10秒後にFalseを返すようにするとか?

どのように解決するのですか?

のドキュメントに、操作をタイムアウトさせる手順が記載されています。 シグナル .

基本的な考え方は、シグナルハンドラを使用して、ある時間間隔のアラームを設定し、そのタイマーが切れた時点で例外を発生させるというものです。

なお、これは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():
    ...