ホームページ >バックエンド開発 >Python チュートリアル >シェルスクリプトで事前定義された期間を持つタイムアウト関数を実装するにはどうすればよいですか?

シェルスクリプトで事前定義された期間を持つタイムアウト関数を実装するにはどうすればよいですか?

Barbara Streisand
Barbara Streisandオリジナル
2024-12-07 11:25:15355ブラウズ

How to Implement a Timeout Function in Shell Scripting with a Predefined Duration?

期間が事前定義されたタイムアウト関数

シェル スクリプトでは、過度の待機を避けるために長時間実行されるタスクを終了するメカニズムがあると便利です。ここで直面する課題は、タイムアウト スクリプト内で関数をラップし、指定された制限時間を超えた場合に False を返すことです。

1 つのアプローチは、事前定義された間隔の後に False 応答をトリガーする非同期タイマーを設定することです。幸いなことに、このような解決策は、signal ライブラリ (UNIX ベースのシステムで利用可能) のシグナル ハンドラーを使用して実現できます。

このプロセスには、シグナルを活用するカスタム デコレーター (@timeout) の作成が含まれます。 alarm() を使用して、希望の時間間隔でアラームを設定します。装飾された関数内で、アラームの有効期限が切れると、TimeoutError 例外が発生し、実質的に操作が中断されます。

このソリューションをコードに組み込むには、次のコードを timeout.py として保存し、 import it:

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

# Short timeout
@timeout(5)
def slow_function():
    # ...

# Default timeout
@timeout
def another_slow_function():
    # ...

# Customize timeout and error message
@timeout(30, error_message="Task timed out")
def yet_another_slow_function():
    # ...

このアプローチにより、関数が指定されたタイムアウト間隔より長くかかる場合、TimeoutError が発生して False が返され、予期しない遅延を適切に処理できるようになります。

以上がシェルスクリプトで事前定義された期間を持つタイムアウト関数を実装するにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。