「労働者が自分の仕事をうまくやりたいなら、まず自分の道具を研ぎ澄まさなければなりません。」 - 孔子、「論語。陸霊公」
表紙 > プログラミング > Pythonで子プロセスの出力をファイルとターミナルに同時にリダイレクトする方法は?

Pythonで子プロセスの出力をファイルとターミナルに同時にリダイレクトする方法は?

2024 年 11 月 8 日に公開
ブラウズ:518

How to Redirect Child Process Output to Files and Terminal Simultaneously in Python?

Pythonで子プロセスの結果をファイルとターミナルに同時に出力する方法

subprocess.call()を使用する場合、次のように指定できます。ファイル記述子を outf および errf として使用して、stdout および stderr を特定のファイルにリダイレクトします。ただし、これらの結果はターミナルに同時に表示されません。

Popen とスレッド化を使用した解決策:

これを克服するには、Popen を直接活用し、 stdout=子プロセスの stdout から読み取る PIPE 引数。方法は次のとおりです:

import subprocess
from threading import Thread

def tee(infile, *files):
    # Forward output from `infile` to `files` in a separate thread
    def fanout(infile, *files):
        for line in iter(infile.readline, ""):
            for f in files:
                f.write(line)

    t = Thread(target=fanout, args=(infile,)   files)
    t.daemon = True
    t.start()
    return t

def teed_call(cmd_args, **kwargs):
    # Override `stdout` and `stderr` arguments with PIPE to capture standard outputs
    stdout, stderr = [kwargs.pop(s, None) for s in ["stdout", "stderr"]]
    p = subprocess.Popen(
        cmd_args,
        stdout=subprocess.PIPE if stdout is not None else None,
        stderr=subprocess.PIPE if stderr is not None else None,
        **kwargs
    )
    
    # Create threads to simultaneously write to files and terminal
    threads = []
    if stdout is not None:
        threads.append(tee(p.stdout, stdout, sys.stdout))
    if stderr is not None:
        threads.append(tee(p.stderr, stderr, sys.stderr))
        
    # Join the threads to ensure IO completion before proceeding
    for t in threads:
        t.join()

    return p.wait()

この関数を使用すると、子プロセスを実行し、その出力をファイルと端末の両方に同時に書き込むことができます:

outf, errf = open("out.txt", "wb"), open("err.txt", "wb")
teed_call(["cat", __file__], stdout=None, stderr=errf)
teed_call(["echo", "abc"], stdout=outf, stderr=errf, bufsize=0)
teed_call(["gcc", "a b"], close_fds=True, stdout=outf, stderr=errf)
最新のチュートリアル もっと>

免責事項: 提供されるすべてのリソースの一部はインターネットからのものです。お客様の著作権またはその他の権利および利益の侵害がある場合は、詳細な理由を説明し、著作権または権利および利益の証拠を提出して、電子メール [email protected] に送信してください。 できるだけ早く対応させていただきます。

Copyright© 2022 湘ICP备2022001581号-3